blob: 5c94455e7680c972a87de960802cd2f58ced9816 [file] [log] [blame]
/*******************************************************************************
* 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();
}
}