/*******************************************************************************
* Copyright (c) 2006, 2016 Mountainminds GmbH & Co. KG and Contributors
* 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:
* Brock Janiczak - initial API and implementation (SF #1774547)
*
******************************************************************************/
package com.mountainminds.eclemma.internal.ui.coverageview;
import org.eclipse.core.runtime.IAdaptable;
import org.eclipse.jdt.core.IClassFile;
import org.eclipse.jdt.core.ICompilationUnit;
import org.eclipse.jdt.core.IJavaElement;
import org.eclipse.jdt.core.JavaModelException;
import org.eclipse.jface.text.IMarkSelection;
import org.eclipse.jface.text.ITextSelection;
import org.eclipse.jface.viewers.ISelection;
import org.eclipse.jface.viewers.IStructuredSelection;
import org.eclipse.jface.viewers.StructuredSelection;
import org.eclipse.jface.viewers.StructuredViewer;
import org.eclipse.ui.IEditorPart;
import org.eclipse.ui.ISelectionListener;
import org.eclipse.ui.IViewPart;
import org.eclipse.ui.IWorkbenchPart;
/**
* Tracks the current selection of Java elements in a workbench page and
* synchronizes the selection of a target view. The selection is either taken
* from a structured viewer of from an editor input.
*/
class SelectionTracker {
private final IViewPart targetview;
private final StructuredViewer target;
private boolean enabled = false;
private IJavaElement currentSelection = null;
private final ISelectionListener listener = new ISelectionListener() {
public void selectionChanged(IWorkbenchPart part, ISelection selection) {
if (part != targetview) {
if (selection instanceof IStructuredSelection) {
IStructuredSelection ssel = (IStructuredSelection) selection;
if (ssel.size() == 1) {
IJavaElement element = getJavaElement(ssel.getFirstElement());
if (element != null)
applySelection(element, false);
}
return;
}
if (part instanceof IEditorPart) {
IJavaElement element = getJavaElement(((IEditorPart) part)
.getEditorInput());
if (element != null) {
element = findElementAtCursor(element, selection);
applySelection(element, false);
}
}
}
}
};
/**
* Try to derive a java element handle from the given object.
*
* @param object
* base object
* @return java element handle or <code>null</code>
*/
private IJavaElement getJavaElement(Object object) {
if (object instanceof IJavaElement) {
return (IJavaElement) object;
}
if (object instanceof IAdaptable) {
IAdaptable a = (IAdaptable) object;
return (IJavaElement) a.getAdapter(IJavaElement.class);
}
return null;
}
/**
* Try to identify a nested java element of the given element from a textual
* selection in its source code. This might be possible if the given element
* is a compilation unit or class file.
*
* @param unit
* unit to search
* @param selection
* selection within this unit
* @return nested element or the unit itself
*/
private IJavaElement findElementAtCursor(IJavaElement unit,
ISelection selection) {
int pos = -1;
if (selection instanceof ITextSelection) {
pos = ((ITextSelection) selection).getOffset();
}
if (selection instanceof IMarkSelection) {
pos = ((IMarkSelection) selection).getOffset();
}
if (pos == -1)
return unit;
IJavaElement element = null;
try {
switch (unit.getElementType()) {
case IJavaElement.COMPILATION_UNIT:
element = ((ICompilationUnit) unit).getElementAt(pos);
break;
case IJavaElement.CLASS_FILE:
element = ((IClassFile) unit).getElementAt(pos);
break;
}
} catch (JavaModelException e) {
// we ignore this
}
return element == null ? unit : element;
}
/**
* Selects the given element in the taget viewer when the tracker is enabled
* and the target view is not active. This conditions can be overruled by the
* <code>force</code> parameter.
*
* @param element
* element to select
* @param force
* if <code>true</code> the selection is applied in any case
*/
private void applySelection(IJavaElement element, boolean force) {
currentSelection = element;
if (force
|| (enabled && targetview != targetview.getSite().getPage()
.getActivePart())) {
target.setSelection(new StructuredSelection(element), true);
}
}
/**
* Creates a new tracker for the given target view and viewer. The tracker
* registers itself with the workbench and must be {@link #dispose()}d when it
* is no longer used.
*
* @param targetview
* view to wich the workbench selections are applied
* @param target
* viewer to wich the workbench selections are applied
*/
public SelectionTracker(IViewPart targetview, StructuredViewer target) {
this.targetview = targetview;
this.target = target;
targetview.getSite().getPage().addPostSelectionListener(listener);
}
/**
* Enables or disables the tracker. If the tracker becomes enabled the last
* workbench selection is immediately applied.
*
* @param enabled
* flag whether the tracker should become enabled
*/
public void setEnabled(boolean enabled) {
this.enabled = enabled;
if (enabled && currentSelection != null) {
applySelection(currentSelection, true);
}
}
/**
* Disposes the tracking functionality. This must be called before the target
* view and viewer get destroyed.
*/
public void dispose() {
targetview.getSite().getPage().removePostSelectionListener(listener);
}
}