blob: afca630a6482c69b7de1b6e2354561579bf82177 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2005 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
*******************************************************************************/
package org.eclipse.jdt.internal.debug.ui.actions;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IAdaptable;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.debug.core.model.IDebugElement;
import org.eclipse.jdt.core.IJavaElement;
import org.eclipse.jdt.core.IType;
import org.eclipse.jdt.core.search.IJavaSearchConstants;
import org.eclipse.jdt.core.search.SearchEngine;
import org.eclipse.jdt.core.search.SearchMatch;
import org.eclipse.jdt.core.search.SearchParticipant;
import org.eclipse.jdt.core.search.SearchPattern;
import org.eclipse.jdt.core.search.SearchRequestor;
import org.eclipse.jdt.debug.core.IJavaArrayType;
import org.eclipse.jdt.debug.core.IJavaType;
import org.eclipse.jdt.debug.ui.IJavaDebugUIConstants;
import org.eclipse.jdt.internal.debug.core.JavaDebugUtils;
import org.eclipse.jdt.internal.debug.ui.JDIDebugUIPlugin;
import org.eclipse.jdt.internal.ui.util.OpenTypeHierarchyUtil;
import org.eclipse.jdt.ui.JavaUI;
import org.eclipse.jface.action.IAction;
import org.eclipse.jface.viewers.IStructuredSelection;
public abstract class OpenTypeAction extends ObjectActionDelegate {
/* (non-Javadoc)
* @see org.eclipse.ui.IActionDelegate#run(org.eclipse.jface.action.IAction)
*/
public void run(IAction action) {
IStructuredSelection selection= getCurrentSelection();
if (selection == null) {
return;
}
Iterator itr= selection.iterator();
try {
while (itr.hasNext()) {
Object element= itr.next();
Object sourceElement = resolveSourceElement(element);
if (sourceElement != null) {
openInEditor(sourceElement);
} else {
IStatus status = new Status(IStatus.INFO, IJavaDebugUIConstants.PLUGIN_ID, IJavaDebugUIConstants.INTERNAL_ERROR, ActionMessages.OpenTypeAction_0, null);
throw new CoreException(status);
}
}
} catch(CoreException e) {
JDIDebugUIPlugin.statusDialog(e.getStatus());
}
}
protected abstract IDebugElement getDebugElement(IAdaptable element);
/**
* Returns the type to open based on the given selected debug element, or <code>null</code>.
*
* @param element selected debug element
* @return the type to open or <code>null</code> if none
* @throws DebugException
*/
protected abstract IJavaType getTypeToOpen(IDebugElement element) throws CoreException;
/**
* Resolves and returns the source element to open or <code>null</code> if none.
*
* @param e selected element to resolve a source element for
* @return the source element to open or <code>null</code> if none
* @throws CoreException
*/
protected Object resolveSourceElement(Object e) throws CoreException {
Object source = null;
IAdaptable element= (IAdaptable) e;
IDebugElement dbgElement= getDebugElement(element);
if (dbgElement != null) {
IJavaType type = getTypeToOpen(dbgElement);
while (type instanceof IJavaArrayType) {
type = ((IJavaArrayType)type).getComponentType();
}
if (type != null) {
source = JavaDebugUtils.resolveType(type);
if (source == null) {
//resort to looking through the workspace projects for the
//type as the source locators failed.
source = findTypeInWorkspace(type.getName());
}
}
}
return source;
}
protected void openInEditor(Object sourceElement) throws CoreException {
if (isHierarchy()) {
if (sourceElement instanceof IJavaElement) {
OpenTypeHierarchyUtil.open((IJavaElement)sourceElement, getWorkbenchWindow());
} else {
typeHierarchyError();
}
} else {
if(sourceElement instanceof IJavaElement) {
JavaUI.openInEditor((IJavaElement) sourceElement);
}
else {
showErrorMessage(ActionMessages.OpenTypeAction_2);
}
}
}
/**
* Returns whether a type hierarchy should be opened.
*
* @return whether a type hierarchy should be opened
*/
protected boolean isHierarchy() {
return false;
}
/**
* Searches for and returns a type with the given name in the workspace,
* or <code>null</code> if none.
*
* @param typeName fully qualified type name
* @return type or <code>null</code>
* @throws JavaModelException
*/
public static IType findTypeInWorkspace(String typeName) throws CoreException {
IType[] types= findTypes(typeName, null);
if (types.length > 0) {
return types[0];
}
return null;
}
private static IType[] findTypes(String typeName, IProgressMonitor monitor) throws CoreException {
final List results= new ArrayList();
SearchRequestor collector= new SearchRequestor() {
public void acceptSearchMatch(SearchMatch match) throws CoreException {
Object element= match.getElement();
if (element instanceof IType)
results.add(element);
}
};
SearchEngine engine= new SearchEngine();
SearchPattern pattern= SearchPattern.createPattern(typeName, IJavaSearchConstants.TYPE, IJavaSearchConstants.DECLARATIONS, SearchPattern.R_EXACT_MATCH);
engine.search(pattern, new SearchParticipant[] { SearchEngine.getDefaultSearchParticipant() }, SearchEngine.createWorkspaceScope(), collector, monitor);
return (IType[]) results.toArray(new IType[results.size()]);
}
protected void typeHierarchyError() {
showErrorMessage(ActionMessages.ObjectActionDelegate_Unable_to_display_type_hierarchy__The_selected_source_element_is_not_contained_in_the_workspace__1);
}
}