ext.org.eclipse.jdt.internal.ui.search.LevelTreeContentProvider.java Source code

Java tutorial

Introduction

Here is the source code for ext.org.eclipse.jdt.internal.ui.search.LevelTreeContentProvider.java

Source

/*******************************************************************************
 * 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 ext.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();
    }

}