| /* |
| * 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.refactoring.util.classMembers; |
| |
| import com.intellij.psi.*; |
| import com.intellij.refactoring.RefactoringBundle; |
| import com.intellij.refactoring.classMembers.MemberDependencyGraph; |
| import com.intellij.refactoring.classMembers.MemberInfoBase; |
| import com.intellij.util.containers.HashMap; |
| |
| import java.util.HashSet; |
| import java.util.Iterator; |
| import java.util.Set; |
| |
| public class InterfaceMemberDependencyGraph<T extends PsiMember, M extends MemberInfoBase<T>> implements MemberDependencyGraph<T, M> { |
| protected HashSet<T> myInterfaceDependencies = null; |
| protected HashMap<T,HashSet<T>> myMembersToInterfacesMap = new HashMap<T, HashSet<T>>(); |
| protected HashSet<PsiClass> myImplementedInterfaces; |
| protected HashMap<PsiClass,HashSet<T>> myMethodsFromInterfaces; |
| protected PsiClass myClass; |
| |
| public InterfaceMemberDependencyGraph(PsiClass aClass) { |
| myClass = aClass; |
| myImplementedInterfaces = new HashSet<PsiClass>(); |
| myMethodsFromInterfaces = new HashMap<PsiClass, HashSet<T>>(); |
| } |
| |
| @Override |
| public void memberChanged(M memberInfo) { |
| if (ClassMembersUtil.isImplementedInterface(memberInfo)) { |
| final PsiClass aClass = (PsiClass) memberInfo.getMember(); |
| myInterfaceDependencies = null; |
| myMembersToInterfacesMap = null; |
| if(memberInfo.isChecked()) { |
| myImplementedInterfaces.add(aClass); |
| } |
| else { |
| myImplementedInterfaces.remove(aClass); |
| } |
| } |
| } |
| |
| @Override |
| public Set<? extends T> getDependent() { |
| if(myInterfaceDependencies == null) { |
| myInterfaceDependencies = new HashSet<T>(); |
| myMembersToInterfacesMap = new HashMap<T, HashSet<T>>(); |
| for (final PsiClass implementedInterface : myImplementedInterfaces) { |
| addInterfaceDeps(implementedInterface); |
| } |
| } |
| return myInterfaceDependencies; |
| } |
| |
| @Override |
| public Set<? extends T> getDependenciesOf(PsiMember member) { |
| final Set dependent = getDependent(); |
| if(dependent.contains(member)) return myMembersToInterfacesMap.get(member); |
| return null; |
| } |
| |
| public String getElementTooltip(PsiMember member) { |
| final Set<? extends PsiMember> dependencies = getDependenciesOf(member); |
| if(dependencies == null || dependencies.size() == 0) return null; |
| StringBuilder buffer = new StringBuilder(); |
| buffer.append(RefactoringBundle.message("interface.member.dependency.required.by.interfaces", dependencies.size())); |
| buffer.append(" "); |
| for (Iterator<? extends PsiMember> iterator = dependencies.iterator(); iterator.hasNext();) { |
| PsiClass aClass = (PsiClass) iterator.next(); |
| buffer.append(aClass.getName()); |
| if(iterator.hasNext()) { |
| buffer.append(", "); |
| } |
| } |
| return buffer.toString(); |
| } |
| |
| protected void addInterfaceDeps(PsiClass intf) { |
| HashSet<T> interfaceMethods = myMethodsFromInterfaces.get(intf); |
| |
| if(interfaceMethods == null) { |
| interfaceMethods = new HashSet<T>(); |
| buildInterfaceMethods(interfaceMethods, intf); |
| myMethodsFromInterfaces.put(intf, interfaceMethods); |
| } |
| for (T method : interfaceMethods) { |
| HashSet<T> interfaces = myMembersToInterfacesMap.get(method); |
| if (interfaces == null) { |
| interfaces = new HashSet<T>(); |
| myMembersToInterfacesMap.put(method, interfaces); |
| } |
| interfaces.add((T)intf); |
| } |
| myInterfaceDependencies.addAll(interfaceMethods); |
| } |
| |
| private void buildInterfaceMethods(HashSet<T> interfaceMethods, PsiClass intf) { |
| PsiMethod[] methods = intf.getMethods(); |
| for (PsiMethod method1 : methods) { |
| PsiMethod method = myClass.findMethodBySignature(method1, true); |
| if (method != null) { |
| interfaceMethods.add((T)method); |
| } |
| } |
| |
| PsiReferenceList implementsList = intf.getImplementsList(); |
| if (implementsList != null) { |
| PsiClassType[] implemented = implementsList.getReferencedTypes(); |
| for (PsiClassType aImplemented : implemented) { |
| PsiClass resolved = aImplemented.resolve(); |
| if (resolved != null) { |
| buildInterfaceMethods(interfaceMethods, resolved); |
| } |
| } |
| } |
| |
| PsiReferenceList extendsList = intf.getExtendsList(); |
| if (extendsList != null) { |
| PsiClassType[] extended = extendsList.getReferencedTypes(); |
| for (PsiClassType aExtended : extended) { |
| PsiClass ref = aExtended.resolve(); |
| if (ref != null) { |
| buildInterfaceMethods(interfaceMethods, ref); |
| } |
| } |
| } |
| } |
| |
| } |