blob: 0df29da2184991223dcd870c55312079924f85d6 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2013 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
* Johann Draschwandtner (Wind River) - [300988] Support filtering variables
*******************************************************************************/
package org.eclipse.debug.ui;
import java.util.ArrayList;
import java.util.Arrays;
import org.eclipse.core.variables.IDynamicVariable;
import org.eclipse.core.variables.IStringVariable;
import org.eclipse.core.variables.VariablesPlugin;
import org.eclipse.debug.internal.core.IInternalDebugCoreConstants;
import org.eclipse.debug.internal.ui.DebugUIPlugin;
import org.eclipse.debug.internal.ui.IDebugHelpContextIds;
import org.eclipse.debug.internal.ui.SWTFactory;
import org.eclipse.debug.internal.ui.preferences.StringVariablePreferencePage;
import org.eclipse.debug.internal.ui.stringsubstitution.StringSubstitutionMessages;
import org.eclipse.debug.internal.ui.stringsubstitution.StringVariableLabelProvider;
import org.eclipse.debug.internal.ui.stringsubstitution.StringVariablePresentationManager;
import org.eclipse.debug.ui.stringsubstitution.IArgumentSelector;
import org.eclipse.jface.dialogs.IDialogSettings;
import org.eclipse.jface.preference.IPreferenceNode;
import org.eclipse.jface.preference.PreferenceDialog;
import org.eclipse.jface.preference.PreferenceManager;
import org.eclipse.jface.preference.PreferenceNode;
import org.eclipse.jface.window.Window;
import org.eclipse.swt.SWT;
import org.eclipse.swt.custom.BusyIndicator;
import org.eclipse.swt.events.SelectionAdapter;
import org.eclipse.swt.events.SelectionEvent;
import org.eclipse.swt.layout.GridData;
import org.eclipse.swt.widgets.Button;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Control;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Label;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.swt.widgets.Text;
import org.eclipse.ui.PlatformUI;
import org.eclipse.ui.dialogs.ElementListSelectionDialog;
/**
* A dialog that prompts the user to choose and configure a string
* substitution variable.
* <p>
* Clients may instantiate this class.
* </p>
* @since 3.1
* @noextend This class is not intended to be subclassed by clients.
*/
public class StringVariableSelectionDialog extends ElementListSelectionDialog {
// button to configure variable's argument
private Button fArgumentButton;
// variable description
private Text fDescriptionText;
// the argument value
private Text fArgumentText;
private String fArgumentValue;
private Button fShowAllButton;
private Label fShowAllDescription;
/**
* Base class for custom variable filters. Clients may extend this class
* to filter specific dynamic variables from the selection dialog.
*
* @since 3.6
*/
public static class VariableFilter {
/**
* Returns whether the given variable should be filtered.
*
* @param var variable to be consider
* @return <code>true</code> to filter the variable, otherwise <code>false</code>
*/
public boolean isFiltered(IDynamicVariable var) {
return false;
}
}
//no filtering by default
private ArrayList<VariableFilter> fFilters = new ArrayList<>();
//when filtering is on, do not show all by default
private boolean fShowAllSelected = false;
/**
* Constructs a new string substitution variable selection dialog.
*
* @param parent parent shell
*/
public StringVariableSelectionDialog(Shell parent) {
super(parent, new StringVariableLabelProvider());
setShellStyle(getShellStyle() | SWT.RESIZE);
setTitle(StringSubstitutionMessages.StringVariableSelectionDialog_2);
setMessage(StringSubstitutionMessages.StringVariableSelectionDialog_3);
setMultipleSelection(false);
setElements(VariablesPlugin.getDefault().getStringVariableManager().getVariables());
}
/**
* Returns the variable expression the user generated from this
* dialog, or <code>null</code> if none.
*
* @return variable expression the user generated from this
* dialog, or <code>null</code> if none
*/
public String getVariableExpression() {
Object[] selected = getResult();
if (selected != null && selected.length == 1) {
IStringVariable variable = (IStringVariable)selected[0];
StringBuffer buffer = new StringBuffer();
buffer.append("${"); //$NON-NLS-1$
buffer.append(variable.getName());
if (fArgumentValue != null && fArgumentValue.length() > 0) {
buffer.append(":"); //$NON-NLS-1$
buffer.append(fArgumentValue);
}
buffer.append("}"); //$NON-NLS-1$
return buffer.toString();
}
return null;
}
/**
* Add the given variable filter. Has no effect if the given filter has
* already been added. Must be called before the dialog is opened.
*
* @param filter the filter to add
* @since 3.6
*/
public void addVariableFilter(VariableFilter filter) {
if(!fFilters.contains(filter)) {
fFilters.add(filter);
}
}
/**
* Sets the filters, replacing any previous filters.
* Must be called before the dialog is opened.
*
* @param filters
* an array of variable filters, use empty Array or <code>null</code> to reset all Filters.
* @since 3.6
*/
public void setFilters(VariableFilter[] filters) {
fFilters.clear();
if(filters != null && filters.length > 0) {
fFilters.addAll(Arrays.asList(filters));
}
}
private void updateElements() {
final Display display = DebugUIPlugin.getStandardDisplay();
BusyIndicator.showWhile(display, new Runnable() {
@Override
public void run() {
final IStringVariable[] elements = VariablesPlugin.getDefault().getStringVariableManager().getVariables();
display.asyncExec(new Runnable() {
@Override
public void run() {
setListElements(elements);
}
});
}
});
}
private void updateDescription() {
if((fShowAllDescription != null) && !fShowAllDescription.isDisposed()) {
if(fShowAllSelected) {
fShowAllDescription.setText(StringSubstitutionMessages.StringVariableSelectionDialog_11);
} else {
fShowAllDescription.setText(StringSubstitutionMessages.StringVariableSelectionDialog_10);
}
}
}
@Override
protected void setListElements(Object[] elements) {
ArrayList<Object> filtered = new ArrayList<>();
filtered.addAll(Arrays.asList(elements));
if(!fFilters.isEmpty() && !fShowAllSelected) {
for (int i = 0; i < elements.length; i++) {
if(elements[i] instanceof IDynamicVariable) {
boolean bFiltered = false;
for (int j = 0; (j < fFilters.size()) && !bFiltered; j++) {
VariableFilter filter = fFilters.get(j);
if(filter.isFiltered((IDynamicVariable)elements[i])) {
filtered.remove(elements[i]);
bFiltered = true;
}
}
}
}
}
super.setListElements(filtered.toArray());
}
/* (non-Javadoc)
* @see org.eclipse.jface.dialogs.Dialog#createContents(org.eclipse.swt.widgets.Composite)
*/
@Override
protected Control createContents(Composite parent) {
Control ctrl = super.createContents(parent);
PlatformUI.getWorkbench().getHelpSystem().setHelp(ctrl, IDebugHelpContextIds.VARIABLE_SELECTION_DIALOG);
return ctrl;
}
/* (non-Javadoc)
* @see org.eclipse.jface.dialogs.Dialog#createDialogArea(org.eclipse.swt.widgets.Composite)
*/
@Override
protected Control createDialogArea(Composite parent) {
Control control = super.createDialogArea(parent);
createArgumentArea((Composite)control);
return control;
}
/**
* Creates an area to display a description of the selected variable
* and a button to configure the variable's argument.
*
* @param parent parent widget
*/
private void createArgumentArea(Composite parent) {
Composite container = SWTFactory.createComposite(parent, parent.getFont(), 2, 1, GridData.FILL_HORIZONTAL, 0, 0);
Composite btnContainer = SWTFactory.createComposite(container, parent.getFont(), 3, 2, GridData.FILL_HORIZONTAL, 0, 0);
boolean bNeedShowAll = false;
if(!fFilters.isEmpty()) {
Object[] elements = VariablesPlugin.getDefault().getStringVariableManager().getVariables();
for (int i = 0;(i < elements.length) && !bNeedShowAll; i++) {
if(elements[i] instanceof IDynamicVariable) {
for (int j = 0; (j < fFilters.size()) && !bNeedShowAll; j++) {
VariableFilter filter = fFilters.get(j);
if(filter.isFiltered((IDynamicVariable)elements[i])) {
bNeedShowAll = true;
}
}
}
}
}
if (bNeedShowAll) {
fShowAllDescription = SWTFactory.createLabel(btnContainer, "", 3); //$NON-NLS-1$
updateDescription();
fShowAllButton = SWTFactory.createCheckButton(btnContainer, StringSubstitutionMessages.StringVariableSelectionDialog_9, null, fShowAllSelected, 1);
fShowAllButton.addSelectionListener(new SelectionAdapter() {
@Override
public void widgetSelected(SelectionEvent e) {
fShowAllSelected = fShowAllButton.getSelection();
updateDescription();
updateElements();
}
});
SWTFactory.createHorizontalSpacer(btnContainer, 1);
} else {
SWTFactory.createHorizontalSpacer(btnContainer, 2);
}
Button editButton = SWTFactory.createPushButton(btnContainer, StringSubstitutionMessages.StringVariableSelectionDialog_0, null, GridData.HORIZONTAL_ALIGN_END);
editButton.addSelectionListener(new SelectionAdapter() {
@Override
public void widgetSelected(SelectionEvent e) {
editVariables();
}
});
SWTFactory.createWrapLabel(container, StringSubstitutionMessages.StringVariableSelectionDialog_6, 2);
Composite args = SWTFactory.createComposite(container, container.getFont(), 2, 2, GridData.FILL_HORIZONTAL, 0, 0);
fArgumentText = new Text(args, SWT.BORDER);
fArgumentText.setFont(container.getFont());
fArgumentText.setLayoutData(new GridData(GridData.FILL_HORIZONTAL));
fArgumentButton = SWTFactory.createPushButton(args, StringSubstitutionMessages.StringVariableSelectionDialog_7, null);
fArgumentButton.addSelectionListener(new SelectionAdapter() {
@Override
public void widgetSelected(SelectionEvent e) {
configureArgument();
}
});
SWTFactory.createWrapLabel(container, StringSubstitutionMessages.StringVariableSelectionDialog_8, 2);
fDescriptionText = new Text(container, SWT.BORDER | SWT.WRAP | SWT.V_SCROLL);
fDescriptionText.setFont(container.getFont());
fDescriptionText.setEditable(false);
GridData gd = new GridData(GridData.FILL_HORIZONTAL);
gd.horizontalSpan = 2;
gd.heightHint = 50;
fDescriptionText.setLayoutData(gd);
}
/**
* Opens the preference dialog to the correct page an allows editing of variables
*/
protected void editVariables() {
final Display display = DebugUIPlugin.getStandardDisplay();
BusyIndicator.showWhile(display, new Runnable() {
@Override
public void run() {
// show the preference page in a new dialog rather than using the utility method to re-use a
// preference page, in case this dialog is being opened from a preference page
if (showVariablesPage()) {
final IStringVariable[] elements = VariablesPlugin.getDefault().getStringVariableManager().getVariables();
display.asyncExec(new Runnable() {
@Override
public void run() {
setListElements(elements);
}
});
}
}
});
}
/**
* Shows the string variables preference page and returns <code>true</code> if OK was pressed.
*
* @return whether OK was pressed
*/
private boolean showVariablesPage() {
StringVariablePreferencePage page = new StringVariablePreferencePage();
page.setTitle(StringSubstitutionMessages.StringVariableSelectionDialog_1);
final IPreferenceNode targetNode = new PreferenceNode("org.eclipse.debug.ui.StringVariablePreferencePage", page); //$NON-NLS-1$
PreferenceManager manager = new PreferenceManager();
manager.addToRoot(targetNode);
final PreferenceDialog dialog = new PreferenceDialog(DebugUIPlugin.getShell(), manager);
final boolean [] result = new boolean[] { false };
BusyIndicator.showWhile(DebugUIPlugin.getStandardDisplay(), new Runnable() {
@Override
public void run() {
dialog.create();
dialog.setMessage(targetNode.getLabelText());
result[0]= (dialog.open() == Window.OK);
}
});
return result[0];
}
/**
* Configures the argument for the selected variable.
*/
protected void configureArgument() {
Object[] objects = getSelectedElements();
IStringVariable variable = (IStringVariable)objects[0];
IArgumentSelector selector = StringVariablePresentationManager.getDefault().getArgumentSelector(variable);
String value = selector.selectArgument(variable, getShell());
if (value != null) {
fArgumentText.setText(value);
}
}
/**
* Update variable description and argument button enablement.
*
* @see org.eclipse.ui.dialogs.AbstractElementListSelectionDialog#handleSelectionChanged()
*/
@Override
protected void handleSelectionChanged() {
super.handleSelectionChanged();
Object[] objects = getSelectedElements();
boolean buttonEnabled = false;
boolean argEnabled = false;
String text = null;
if (objects.length == 1) {
IStringVariable variable = (IStringVariable)objects[0];
IArgumentSelector selector = StringVariablePresentationManager.getDefault().getArgumentSelector(variable);
if (variable instanceof IDynamicVariable) {
argEnabled = ((IDynamicVariable)variable).supportsArgument();
}
buttonEnabled = argEnabled && selector != null;
text = variable.getDescription();
}
if (text == null) {
text = IInternalDebugCoreConstants.EMPTY_STRING;
}
fArgumentText.setEnabled(argEnabled);
fArgumentButton.setEnabled(buttonEnabled);
fDescriptionText.setText(text);
}
/* (non-Javadoc)
* @see org.eclipse.jface.dialogs.Dialog#okPressed()
*/
@Override
protected void okPressed() {
fArgumentValue = fArgumentText.getText().trim();
super.okPressed();
}
/**
* Returns the name of the section that this dialog stores its settings in
*
* @return String
*/
private String getDialogSettingsSectionName() {
return IDebugUIConstants.PLUGIN_ID + ".STRING_VARIABLE_SELECTION_DIALOG_SECTION"; //$NON-NLS-1$
}
/* (non-Javadoc)
* @see org.eclipse.jface.dialogs.Dialog#getDialogBoundsSettings()
*/
@Override
protected IDialogSettings getDialogBoundsSettings() {
IDialogSettings settings = DebugUIPlugin.getDefault().getDialogSettings();
IDialogSettings section = settings.getSection(getDialogSettingsSectionName());
if (section == null) {
section = settings.addNewSection(getDialogSettingsSectionName());
}
return section;
}
}