blob: f4fa26b2511e7e053420904f343003f491960f15 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2012 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
* Samrat Dhillon samrat.dhillon@gmail.com - Bug 356070 Step into selection does not work for generic types
*******************************************************************************/
package org.eclipse.jdt.internal.debug.ui.actions;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IAdapterManager;
import org.eclipse.core.runtime.Platform;
import org.eclipse.debug.core.DebugEvent;
import org.eclipse.debug.core.DebugException;
import org.eclipse.debug.core.DebugPlugin;
import org.eclipse.debug.core.IDebugEventSetListener;
import org.eclipse.debug.core.model.IStackFrame;
import org.eclipse.debug.core.model.IThread;
import org.eclipse.debug.ui.actions.IRunToLineTarget;
import org.eclipse.jdt.core.IJavaElement;
import org.eclipse.jdt.core.IMember;
import org.eclipse.jdt.core.IMethod;
import org.eclipse.jdt.core.IType;
import org.eclipse.jdt.core.JavaModelException;
import org.eclipse.jdt.debug.core.IJavaStackFrame;
import org.eclipse.jdt.debug.core.IJavaThread;
import org.eclipse.jdt.internal.debug.ui.EvaluationContextManager;
import org.eclipse.jdt.internal.debug.ui.JDIDebugUIPlugin;
import org.eclipse.jface.action.IAction;
import org.eclipse.jface.text.IDocument;
import org.eclipse.jface.text.IRegion;
import org.eclipse.jface.text.ITextSelection;
import org.eclipse.jface.text.TextSelection;
import org.eclipse.jface.viewers.ISelection;
import org.eclipse.ui.IEditorActionDelegate;
import org.eclipse.ui.IEditorPart;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.IWorkbenchWindowActionDelegate;
import org.eclipse.ui.texteditor.IEditorStatusLine;
import org.eclipse.ui.texteditor.ITextEditor;
/**
* Steps into the selected method.
*/
public class StepIntoSelectionActionDelegate implements IEditorActionDelegate, IWorkbenchWindowActionDelegate {
private IEditorPart fEditorPart = null;
private IWorkbenchWindow fWindow = null;
private IRegion fTextRegion = null;
/**
* Default constructor
*/
public StepIntoSelectionActionDelegate() {}
/**
* Constructor
* @param region
*/
public StepIntoSelectionActionDelegate(IRegion region) {
fTextRegion = region;
}
/**
* The name of the type being "run to".
* @see StepIntoSelectionActionDelegate#runToLineBeforeStepIn(ITextSelection, IJavaStackFrame, IMethod)
*/
private String fRunToLineType= null;
/**
* The line number being "run to."
* @see StepIntoSelectionActionDelegate#runToLineBeforeStepIn(ITextSelection, IJavaStackFrame, IMethod)
*/
private int fRunToLineLine= -1;
/**
* The debug event list listener used to know when a run to line has finished.
* @see StepIntoSelectionActionDelegate#runToLineBeforeStepIn(ITextSelection, IJavaStackFrame, IMethod)
*/
private IDebugEventSetListener listener= null;
/**
* @see org.eclipse.ui.IActionDelegate#run(org.eclipse.jface.action.IAction)
*/
public void run(IAction action) {
IJavaStackFrame frame = getStackFrame();
if (frame == null || !frame.isSuspended()) {
// no longer suspended - unexpected
return;
}
ITextSelection textSelection = getTextSelection();
try {
IEditorPart activeEditor = getActiveEditor();
IJavaElement javaElement= StepIntoSelectionUtils.getJavaElement(activeEditor.getEditorInput());
IMethod method = StepIntoSelectionUtils.getMethod(textSelection, javaElement);
if (method == null) {
method = StepIntoSelectionUtils.getFirstMethodOnLine(textSelection.getOffset(), activeEditor, javaElement);
}
IType callingType = getType(textSelection);
if (method == null || callingType == null) {
return;
}
int lineNumber = frame.getLineNumber();
String callingTypeName = stripInnerNamesAndParameterType(callingType.getFullyQualifiedName());
String frameName = stripInnerNamesAndParameterType(frame.getDeclaringTypeName());
// debug line numbers are 1 based, document line numbers are 0 based
if (textSelection.getStartLine() == (lineNumber - 1) && callingTypeName.equals(frameName)) {
doStepIn(frame, method);
} else {
// not on current line
runToLineBeforeStepIn(textSelection, frame.getThread(), method);
return;
}
}
catch (DebugException e) {
showErrorMessage(e.getStatus().getMessage());
return;
}
catch(JavaModelException jme) {
showErrorMessage(jme.getStatus().getMessage());
return;
}
}
/**
* Steps into the given method in the given stack frame
* @param frame the frame in which the step should begin
* @param method the method to step into
* @throws DebugException
*/
private void doStepIn(IJavaStackFrame frame, IMethod method) throws DebugException {
// ensure top stack frame
IStackFrame tos = frame.getThread().getTopStackFrame();
if (tos == null) {
return;
}
if (!tos.equals(frame)) {
showErrorMessage(ActionMessages.StepIntoSelectionActionDelegate_Step_into_selection_only_available_in_top_stack_frame__3);
return;
}
StepIntoSelectionHandler handler = new StepIntoSelectionHandler((IJavaThread)frame.getThread(), frame, method);
handler.step();
}
/**
* When the user chooses to "step into selection" on a line other than
* the currently executing one, first perform a "run to line" to get to
* the desired location, then perform a "step into selection."
*/
private void runToLineBeforeStepIn(ITextSelection textSelection, final IThread thread, final IMethod method) {
fRunToLineType = getType(textSelection).getFullyQualifiedName();
fRunToLineLine = textSelection.getStartLine() + 1;
if (fRunToLineType == null || fRunToLineLine == -1) {
return;
}
// see bug 65489 - get the run-to-line adapter from the editor
IRunToLineTarget runToLineAction = null;
IEditorPart ed = getActiveEditor();
if (ed != null) {
runToLineAction = (IRunToLineTarget) ed.getAdapter(IRunToLineTarget.class);
if (runToLineAction == null) {
IAdapterManager adapterManager = Platform.getAdapterManager();
if (adapterManager.hasAdapter(ed, IRunToLineTarget.class.getName())) {
runToLineAction = (IRunToLineTarget) adapterManager.loadAdapter(ed,IRunToLineTarget.class.getName());
}
}
}
// if no adapter exists, use the Java adapter
if (runToLineAction == null) {
runToLineAction = new RunToLineAdapter();
}
listener= new IDebugEventSetListener() {
/**
* @see IDebugEventSetListener#handleDebugEvents(DebugEvent[])
*/
public void handleDebugEvents(DebugEvent[] events) {
for (int i = 0; i < events.length; i++) {
DebugEvent event = events[i];
switch (event.getKind()) {
case DebugEvent.SUSPEND :
handleSuspendEvent(event);
break;
case DebugEvent.TERMINATE :
handleTerminateEvent(event);
break;
default :
break;
}
}
}
/**
* Listen for the completion of the "run to line." When the thread
* suspends at the correct location, perform a "step into selection"
* @param event the debug event
*/
private void handleSuspendEvent(DebugEvent event) {
Object source = event.getSource();
if (source instanceof IJavaThread) {
try {
final IJavaStackFrame frame= (IJavaStackFrame) ((IJavaThread) source).getTopStackFrame();
if (isExpectedFrame(frame)) {
DebugPlugin plugin = DebugPlugin.getDefault();
plugin.removeDebugEventListener(listener);
plugin.asyncExec(new Runnable() {
public void run() {
try {
doStepIn(frame, method);
} catch (DebugException e) {
showErrorMessage(e.getStatus().getMessage());
}
}
});
}
} catch (DebugException e) {
return;
}
}
}
/**
* Returns whether the given frame is the frame that this action is expecting.
* This frame is expecting a stack frame for the suspension of the "run to line".
* @param frame the given stack frame or <code>null</code>
* @return whether the given stack frame is the expected frame
* @throws DebugException
*/
private boolean isExpectedFrame(IJavaStackFrame frame) throws DebugException {
return frame != null &&
fRunToLineLine == frame.getLineNumber() &&
frame.getReceivingTypeName().equals(fRunToLineType);
}
/**
* When the debug target we're listening for terminates, stop listening
* to debug events.
* @param event the debug event
*/
private void handleTerminateEvent(DebugEvent event) {
Object source = event.getSource();
if (thread.getDebugTarget() == source) {
DebugPlugin.getDefault().removeDebugEventListener(listener);
}
}
};
DebugPlugin.getDefault().addDebugEventListener(listener);
try {
runToLineAction.runToLine(getActiveEditor(), textSelection, thread);
} catch (CoreException e) {
DebugPlugin.getDefault().removeDebugEventListener(listener);
showErrorMessage(ActionMessages.StepIntoSelectionActionDelegate_4);
JDIDebugUIPlugin.log(e.getStatus());
}
}
/**
* Gets the current text selection from the currently active editor
* @return the current text selection
*/
private ITextSelection getTextSelection() {
IEditorPart part = getActiveEditor();
if (part instanceof ITextEditor) {
ITextEditor editor = (ITextEditor)part;
if(fTextRegion != null) {
IDocument document = editor.getDocumentProvider().getDocument(editor.getEditorInput());
if(document != null) {
return new TextSelection(document, fTextRegion.getOffset(), fTextRegion.getLength());
}
}
else {
return (ITextSelection)editor.getSelectionProvider().getSelection();
}
}
showErrorMessage(ActionMessages.StepIntoSelectionActionDelegate_Step_into_selection_only_available_in_Java_editor__4);
return null;
}
/**
* Return the type containing the selected text, or <code>null</code> if the
* selection is not in a type.
*/
protected IType getType(ITextSelection textSelection) {
IMember member= ActionDelegateHelper.getDefault().getCurrentMember(textSelection);
IType type= null;
if (member instanceof IType) {
type = (IType)member;
} else if (member != null) {
type= member.getDeclaringType();
}
return type;
}
/**
* Displays an error message in the status area
*
* @param message
*/
protected void showErrorMessage(String message) {
if (getActiveEditor() != null) {
IEditorStatusLine statusLine= (IEditorStatusLine) getActiveEditor().getAdapter(IEditorStatusLine.class);
if (statusLine != null) {
statusLine.setMessage(true, message, null);
}
}
JDIDebugUIPlugin.getStandardDisplay().beep();
}
/**
* @see org.eclipse.ui.IEditorActionDelegate#setActiveEditor(org.eclipse.jface.action.IAction, org.eclipse.ui.IEditorPart)
*/
public void setActiveEditor(IAction action, IEditorPart targetEditor) {
fEditorPart = targetEditor;
}
/**
* Returns the active editor or <code>null</code>.
*
* @return active editor or <code>null</code>
*/
protected IEditorPart getActiveEditor() {
if (fWindow != null) {
// global action
return fWindow.getActivePage().getActiveEditor();
}
// pop-up action
return fEditorPart;
}
/**
* Strips inner class names and parameterized type information from the given type name.
*
* @param fullyQualifiedName
*/
private String stripInnerNamesAndParameterType(String fullyQualifiedName) {
// ignore inner class qualification, as the compiler generated names and java model names can be different
String sig = fullyQualifiedName;
int index = sig.indexOf('$');
if (index > 0) {
sig = sig.substring(0, index);
}
//also ignore erasure
index = sig.indexOf('<');
if (index > 0){
sig = sig.substring(0, index);
}
return sig;
}
/**
* @see org.eclipse.ui.IActionDelegate#selectionChanged(org.eclipse.jface.action.IAction, org.eclipse.jface.viewers.ISelection)
*/
public void selectionChanged(IAction action, ISelection selection) {
}
/**
* Returns the current stack frame context, or <code>null</code> if none.
*
* @return the current stack frame context, or <code>null</code> if none
*/
protected IJavaStackFrame getStackFrame() {
return EvaluationContextManager.getEvaluationContext(getActiveEditor());
}
/**
* @see org.eclipse.ui.IWorkbenchWindowActionDelegate#dispose()
*/
public void dispose() {}
/**
* @see org.eclipse.ui.IWorkbenchWindowActionDelegate#init(org.eclipse.ui.IWorkbenchWindow)
*/
public void init(IWorkbenchWindow window) {
fWindow = window;
}
}