blob: ff65059905204c823f57c8ba1132f8a0392d2c93 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2009 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.Iterator;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IAdaptable;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.jdt.core.IJavaElement;
import org.eclipse.jdt.core.IType;
import org.eclipse.jdt.core.JavaModelException;
import org.eclipse.jdt.core.search.IJavaSearchConstants;
import org.eclipse.jdt.core.search.SearchEngine;
import org.eclipse.jdt.core.search.TypeNameMatch;
import org.eclipse.jdt.core.search.TypeNameMatchRequestor;
import org.eclipse.jdt.debug.core.IJavaArrayType;
import org.eclipse.jdt.debug.core.IJavaType;
import org.eclipse.jdt.debug.ui.IJavaDebugUIConstants;
import org.eclipse.jface.action.IAction;
import org.eclipse.jface.viewers.IStructuredSelection;
import org.eclipse.debug.core.DebugException;
import org.eclipse.debug.core.model.IDebugElement;
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;
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, "Source not found", null); //$NON-NLS-1$
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 {
int dot= typeName.lastIndexOf('.');
char[][] qualifications;
String simpleName;
if (dot != -1) {
qualifications= new char[][] { typeName.substring(0, dot).toCharArray() };
simpleName= typeName.substring(dot + 1);
} else {
qualifications= null;
simpleName= typeName;
}
char[][] typeNames= new char[][] { simpleName.toCharArray() };
class ResultException extends RuntimeException {
private static final long serialVersionUID= 1L;
private final IType fType;
public ResultException(IType type) {
fType= type;
}
}
TypeNameMatchRequestor requestor= new TypeNameMatchRequestor() {
public void acceptTypeNameMatch(TypeNameMatch match) {
throw new ResultException(match.getType());
}
};
try {
new SearchEngine().searchAllTypeNames(qualifications, typeNames, SearchEngine.createWorkspaceScope(), requestor, IJavaSearchConstants.WAIT_UNTIL_READY_TO_SEARCH, null);
} catch (ResultException e) {
return e.fType;
}
return null;
}
protected void typeHierarchyError() {
showErrorMessage(ActionMessages.ObjectActionDelegate_Unable_to_display_type_hierarchy__The_selected_source_element_is_not_contained_in_the_workspace__1);
}
}