| /******************************************************************************* |
| * Copyright (c) 2000, 2011 IBM Corporation and others. |
| * All rights reserved. This program and the accompanying materials |
| * are made available under the terms of the Eclipse Public License v1.0 |
| * which accompanies this distribution, and is available at |
| * http://www.eclipse.org/legal/epl-v10.html |
| * |
| * Contributors: |
| * IBM Corporation - initial API and implementation |
| * Carsten Pfeiffer <carsten.pfeiffer@gebit.de> - [search] Custom search results not shown hierarchically in the java search results view - https://bugs.eclipse.org/bugs/show_bug.cgi?id=303705 |
| *******************************************************************************/ |
| |
| package org.eclipse.jdt.internal.ui.search; |
| |
| import java.util.HashMap; |
| import java.util.HashSet; |
| import java.util.Iterator; |
| import java.util.Map; |
| import java.util.Set; |
| |
| import org.eclipse.core.runtime.IAdaptable; |
| |
| import org.eclipse.core.resources.IResource; |
| |
| import org.eclipse.jface.viewers.AbstractTreeViewer; |
| import org.eclipse.jface.viewers.ITreeContentProvider; |
| |
| import org.eclipse.search.ui.text.AbstractTextSearchResult; |
| |
| import org.eclipse.jdt.core.IClassFile; |
| import org.eclipse.jdt.core.ICompilationUnit; |
| import org.eclipse.jdt.core.IJavaElement; |
| import org.eclipse.jdt.core.IType; |
| |
| import org.eclipse.jdt.ui.StandardJavaElementContentProvider; |
| |
| public class LevelTreeContentProvider extends JavaSearchContentProvider implements ITreeContentProvider { |
| private Map<Object, Set<Object>> fChildrenMap; |
| private StandardJavaElementContentProvider fContentProvider; |
| |
| public static final int LEVEL_TYPE= 1; |
| public static final int LEVEL_FILE= 2; |
| public static final int LEVEL_PACKAGE= 3; |
| public static final int LEVEL_PROJECT= 4; |
| |
| private static final int[][] JAVA_ELEMENT_TYPES= {{IJavaElement.TYPE}, |
| {IJavaElement.CLASS_FILE, IJavaElement.COMPILATION_UNIT}, |
| {IJavaElement.PACKAGE_FRAGMENT}, |
| {IJavaElement.JAVA_PROJECT, IJavaElement.PACKAGE_FRAGMENT_ROOT}, |
| {IJavaElement.JAVA_MODEL}}; |
| private static final int[][] RESOURCE_TYPES= { |
| {}, |
| {IResource.FILE}, |
| {IResource.FOLDER}, |
| {IResource.PROJECT}, |
| {IResource.ROOT}}; |
| |
| private static final int MAX_LEVEL= JAVA_ELEMENT_TYPES.length - 1; |
| private int fCurrentLevel; |
| static class FastJavaElementProvider extends StandardJavaElementContentProvider { |
| @Override |
| public Object getParent(Object element) { |
| Object parent= internalGetParent(element); |
| if (parent == null && element instanceof IAdaptable) { |
| IAdaptable adaptable = (IAdaptable)element; |
| Object javaElement= adaptable.getAdapter(IJavaElement.class); |
| if (javaElement != null) { |
| parent= internalGetParent(javaElement); |
| } else { |
| Object resource= adaptable.getAdapter(IResource.class); |
| if (resource != null) { |
| parent= internalGetParent(resource); |
| } |
| } |
| } |
| return parent; |
| } |
| } |
| |
| public LevelTreeContentProvider(JavaSearchResultPage page, int level) { |
| super(page); |
| fCurrentLevel= level; |
| fContentProvider= new FastJavaElementProvider(); |
| } |
| |
| public Object getParent(Object child) { |
| Object possibleParent= internalGetParent(child); |
| if (possibleParent instanceof IJavaElement) { |
| IJavaElement javaElement= (IJavaElement) possibleParent; |
| for (int j= fCurrentLevel; j < MAX_LEVEL + 1; j++) { |
| for (int i= 0; i < JAVA_ELEMENT_TYPES[j].length; i++) { |
| if (javaElement.getElementType() == JAVA_ELEMENT_TYPES[j][i]) { |
| return null; |
| } |
| } |
| } |
| } else if (possibleParent instanceof IResource) { |
| IResource resource= (IResource) possibleParent; |
| for (int j= fCurrentLevel; j < MAX_LEVEL + 1; j++) { |
| for (int i= 0; i < RESOURCE_TYPES[j].length; i++) { |
| if (resource.getType() == RESOURCE_TYPES[j][i]) { |
| return null; |
| } |
| } |
| } |
| } |
| if (fCurrentLevel != LEVEL_FILE && child instanceof IType) { |
| IType type= (IType) child; |
| if (possibleParent instanceof ICompilationUnit |
| || possibleParent instanceof IClassFile) |
| possibleParent= type.getPackageFragment(); |
| } |
| return possibleParent; |
| } |
| |
| private Object internalGetParent(Object child) { |
| return fContentProvider.getParent(child); |
| } |
| |
| public Object[] getElements(Object inputElement) { |
| return getChildren(inputElement); |
| } |
| |
| @Override |
| protected synchronized void initialize(AbstractTextSearchResult result) { |
| super.initialize(result); |
| fChildrenMap= new HashMap<Object, Set<Object>>(); |
| if (result != null) { |
| Object[] elements= result.getElements(); |
| for (int i= 0; i < elements.length; i++) { |
| if (getPage().getDisplayedMatchCount(elements[i]) > 0) { |
| insert(null, null, elements[i]); |
| } |
| } |
| } |
| } |
| |
| protected void insert(Map<Object, Set<Object>> toAdd, Set<Object> toUpdate, Object child) { |
| Object parent= getParent(child); |
| while (parent != null) { |
| if (insertChild(parent, child)) { |
| if (toAdd != null) |
| insertInto(parent, child, toAdd); |
| } else { |
| if (toUpdate != null) |
| toUpdate.add(parent); |
| return; |
| } |
| child= parent; |
| parent= getParent(child); |
| } |
| if (insertChild(getSearchResult(), child)) { |
| if (toAdd != null) |
| insertInto(getSearchResult(), child, toAdd); |
| } |
| } |
| |
| private boolean insertChild(Object parent, Object child) { |
| return insertInto(parent, child, fChildrenMap); |
| } |
| |
| private boolean insertInto(Object parent, Object child, Map<Object, Set<Object>> map) { |
| Set<Object> children= map.get(parent); |
| if (children == null) { |
| children= new HashSet<Object>(); |
| map.put(parent, children); |
| } |
| return children.add(child); |
| } |
| |
| protected void remove(Set<Object> toRemove, Set<Object> toUpdate, Object element) { |
| // precondition here: fResult.getMatchCount(child) <= 0 |
| |
| if (hasChildren(element)) { |
| if (toUpdate != null) |
| toUpdate.add(element); |
| } else { |
| if (getPage().getDisplayedMatchCount(element) == 0) { |
| fChildrenMap.remove(element); |
| Object parent= getParent(element); |
| if (parent != null) { |
| if (removeFromSiblings(element, parent)) { |
| remove(toRemove, toUpdate, parent); |
| } |
| } else { |
| if (removeFromSiblings(element, getSearchResult())) { |
| if (toRemove != null) |
| toRemove.add(element); |
| } |
| } |
| } else { |
| if (toUpdate != null) { |
| toUpdate.add(element); |
| } |
| } |
| } |
| } |
| |
| /** |
| * Tries to remove the given element from the list of stored siblings. |
| * |
| * @param element potential child |
| * @param parent potential parent |
| * @return returns true if it really was a remove (i.e. element was a child of parent). |
| */ |
| private boolean removeFromSiblings(Object element, Object parent) { |
| Set<Object> siblings= fChildrenMap.get(parent); |
| if (siblings != null) { |
| return siblings.remove(element); |
| } else { |
| return false; |
| } |
| } |
| |
| public Object[] getChildren(Object parentElement) { |
| Set<Object> children= fChildrenMap.get(parentElement); |
| if (children == null) |
| return EMPTY_ARR; |
| int limit= getPage().getElementLimit().intValue(); |
| if (limit != -1 && limit < children.size()) { |
| Object[] limitedArray= new Object[limit]; |
| Iterator<Object> iterator= children.iterator(); |
| for (int i= 0; i < limit; i++) { |
| limitedArray[i]= iterator.next(); |
| } |
| return limitedArray; |
| } |
| |
| return children.toArray(); |
| } |
| |
| public boolean hasChildren(Object element) { |
| Set<Object> children= fChildrenMap.get(element); |
| return children != null && !children.isEmpty(); |
| } |
| |
| @Override |
| public synchronized void elementsChanged(Object[] updatedElements) { |
| if (getSearchResult() == null) |
| return; |
| |
| AbstractTreeViewer viewer= (AbstractTreeViewer) getPage().getViewer(); |
| |
| Set<Object> toRemove= new HashSet<Object>(); |
| Set<Object> toUpdate= new HashSet<Object>(); |
| Map<Object, Set<Object>> toAdd= new HashMap<Object, Set<Object>>(); |
| for (int i= 0; i < updatedElements.length; i++) { |
| if (getPage().getDisplayedMatchCount(updatedElements[i]) > 0) |
| insert(toAdd, toUpdate, updatedElements[i]); |
| else |
| remove(toRemove, toUpdate, updatedElements[i]); |
| } |
| |
| viewer.remove(toRemove.toArray()); |
| for (Iterator<Object> iter= toAdd.keySet().iterator(); iter.hasNext();) { |
| Object parent= iter.next(); |
| HashSet<Object> children= (HashSet<Object>) toAdd.get(parent); |
| viewer.add(parent, children.toArray()); |
| } |
| for (Iterator<Object> elementsToUpdate= toUpdate.iterator(); elementsToUpdate.hasNext();) { |
| viewer.refresh(elementsToUpdate.next()); |
| } |
| |
| } |
| |
| @Override |
| public void clear() { |
| initialize(getSearchResult()); |
| getPage().getViewer().refresh(); |
| } |
| |
| public void setLevel(int level) { |
| fCurrentLevel= level; |
| initialize(getSearchResult()); |
| getPage().getViewer().refresh(); |
| } |
| |
| } |