/******************************************************************************* * Copyright (c) 2000, 2017 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 * *******************************************************************************/ package org.eclipse.dltk.internal.ui.callhierarchy; import java.lang.reflect.InvocationTargetException; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.dltk.core.IModelElement; import org.eclipse.dltk.internal.corext.callhierarchy.MethodWrapper; import org.eclipse.dltk.ui.DLTKUIPlugin; import org.eclipse.dltk.ui.util.ExceptionHandler; import org.eclipse.jface.operation.IRunnableContext; import org.eclipse.jface.operation.IRunnableWithProgress; import org.eclipse.jface.viewers.AbstractTreeViewer; import org.eclipse.jface.viewers.ITreeContentProvider; import org.eclipse.jface.viewers.Viewer; import org.eclipse.ui.progress.DeferredTreeContentManager; public class CallHierarchyContentProvider implements ITreeContentProvider { private final static Object[] EMPTY_ARRAY = new Object[0]; private DeferredTreeContentManager fManager; private CallHierarchyViewPart fPart; private class MethodWrapperRunnable implements IRunnableWithProgress { private MethodWrapper fMethodWrapper; private MethodWrapper[] fCalls= null; MethodWrapperRunnable(MethodWrapper methodWrapper) { fMethodWrapper= methodWrapper; } @Override public void run(IProgressMonitor pm) { fCalls= fMethodWrapper.getCalls(pm); } MethodWrapper[] getCalls() { if (fCalls != null) { return fCalls; } return new MethodWrapper[0]; } } public CallHierarchyContentProvider(CallHierarchyViewPart part) { super(); fPart= part; } @Override public Object[] getChildren(Object parentElement) { if (parentElement instanceof TreeRoot) { TreeRoot dummyRoot = (TreeRoot) parentElement; return new Object[] { dummyRoot.getRoot() }; } else if (parentElement instanceof MethodWrapper) { MethodWrapper methodWrapper = ((MethodWrapper) parentElement); if (shouldStopTraversion(methodWrapper)) { return EMPTY_ARRAY; } else { if (fManager != null) { Object[] children = fManager.getChildren(new DeferredMethodWrapper(this, methodWrapper)); if (children != null) return children; } return fetchChildren(methodWrapper); } } return EMPTY_ARRAY; } protected Object[] fetchChildren(MethodWrapper methodWrapper) { IRunnableContext context= DLTKUIPlugin.getActiveWorkbenchWindow(); MethodWrapperRunnable runnable= new MethodWrapperRunnable(methodWrapper); try { context.run(true, true, runnable); } catch (InvocationTargetException e) { ExceptionHandler.handle(e, CallHierarchyMessages.CallHierarchyContentProvider_searchError_title, CallHierarchyMessages.CallHierarchyContentProvider_searchError_message); return EMPTY_ARRAY; } catch (InterruptedException e) { return new Object[] { TreeTermination.SEARCH_CANCELED }; } return runnable.getCalls(); } private boolean shouldStopTraversion(MethodWrapper methodWrapper) { return (methodWrapper.getLevel() > CallHierarchyUI.getDefault().getMaxCallDepth()) || methodWrapper.isRecursive(); } /** * @see org.eclipse.jface.viewers.IStructuredContentProvider#getElements(java.lang.Object) */ @Override public Object[] getElements(Object inputElement) { return getChildren(inputElement); } /** * @see org.eclipse.jface.viewers.ITreeContentProvider#getParent(java.lang.Object) */ @Override public Object getParent(Object element) { if (element instanceof MethodWrapper) { return ((MethodWrapper) element).getParent(); } return null; } /** * @see org.eclipse.jface.viewers.IContentProvider#dispose() */ @Override public void dispose() { // Nothing to dispose } /** * @see org.eclipse.jface.viewers.ITreeContentProvider#hasChildren(java.lang.Object) */ @Override public boolean hasChildren(Object element) { if (element == TreeRoot.EMPTY_ROOT || element == TreeTermination.SEARCH_CANCELED) { return false; } // Only methods can have subelements, so there's no need to fool the // user into believing that there is more if (element instanceof MethodWrapper) { MethodWrapper methodWrapper= (MethodWrapper) element; if (methodWrapper.getMember().getElementType() != IModelElement.METHOD) { return false; } if (shouldStopTraversion(methodWrapper)) { return false; } return true; } else if (element instanceof TreeRoot) { return true; } else if (element instanceof DeferredMethodWrapper) { // Err on the safe side by returning true even though // we don't know for sure that there are children. return true; } return false; // the "Update ..." placeholder has no children } @Override public void inputChanged(Viewer viewer, Object oldInput, Object newInput) { if (oldInput instanceof TreeRoot) { Object root = ((TreeRoot) oldInput).getRoot(); if (root instanceof MethodWrapper) { cancelJobs((MethodWrapper) root); } } if (viewer instanceof AbstractTreeViewer) { fManager = new DeferredTreeContentManager( (AbstractTreeViewer) viewer, fPart.getSite()); } } /** * Cancel all current jobs. */ void cancelJobs(MethodWrapper wrapper) { if (fManager != null && wrapper != null) { fManager.cancel(wrapper); if (fPart != null) { fPart.setCancelEnabled(false); } } } /** * */ public void doneFetching() { if (fPart != null) { fPart.setCancelEnabled(false); } } /** * */ public void startFetching() { if (fPart != null) { fPart.setCancelEnabled(true); } } }