/******************************************************************************* * 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. *

* Clients may instantiate this class. *

* @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 true to filter the variable, otherwise false */ public boolean isFiltered(IDynamicVariable var) { return false; } } //no filtering by default private ArrayList 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 null if none. * * @return variable expression the user generated from this * dialog, or null 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 null 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 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 true 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; } }