From f07f8827305189c63d4841983ba7a1104d8ec714 Mon Sep 17 00:00:00 2001 From: Sudhindra Date: Fri, 6 Oct 2023 14:20:30 +0530 Subject: [PATCH] Run As Context Menu shows Ant Run Configurations multiple times #527 AntLaunchShortcutWithDialog now implements only ILaunchShortcut interface. The common implementation of AntLaunchShortcut and AntLaunchShortcutWithDialog are moved to AntLaunchShortcutCommon class Fixes https://github.com/eclipse-platform/eclipse.platform/issues/527 --- .../AntLaunchShortcut.java | 495 +--------------- .../AntLaunchShortcutCommon.java | 530 ++++++++++++++++++ .../AntLaunchShortcutWithDialog.java | 3 +- 3 files changed, 534 insertions(+), 494 deletions(-) create mode 100644 ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcutCommon.java diff --git a/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcut.java b/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcut.java index f41af133873..e40d547b6fa 100644 --- a/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcut.java +++ b/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcut.java @@ -11,523 +11,32 @@ * Contributors: * IBM Corporation - initial API and implementation * John-Mason P. Shackelford - bug 34548 + * Sudhindra Kulkarni (ETAS GmbH) - Issue 527 *******************************************************************************/ package org.eclipse.ant.internal.ui.launchConfigurations; -import java.text.MessageFormat; -import java.util.ArrayList; -import java.util.Collections; import java.util.List; -import org.eclipse.ant.internal.core.IAntCoreConstants; -import org.eclipse.ant.internal.ui.AntUIPlugin; import org.eclipse.ant.internal.ui.AntUtil; -import org.eclipse.ant.internal.ui.IAntUIConstants; import org.eclipse.ant.internal.ui.model.AntElementNode; -import org.eclipse.ant.internal.ui.model.AntProjectNode; -import org.eclipse.ant.internal.ui.model.AntTargetNode; -import org.eclipse.ant.internal.ui.model.AntTaskNode; -import org.eclipse.ant.launching.IAntLaunchConstants; -import org.eclipse.core.externaltools.internal.IExternalToolConstants; import org.eclipse.core.resources.IContainer; import org.eclipse.core.resources.IFile; -import org.eclipse.core.resources.IProject; import org.eclipse.core.resources.IResource; -import org.eclipse.core.runtime.CoreException; import org.eclipse.core.runtime.IAdaptable; import org.eclipse.core.runtime.IPath; -import org.eclipse.core.runtime.IStatus; -import org.eclipse.core.runtime.Status; -import org.eclipse.core.variables.VariablesPlugin; -import org.eclipse.debug.core.DebugPlugin; import org.eclipse.debug.core.ILaunchConfiguration; -import org.eclipse.debug.core.ILaunchConfigurationType; -import org.eclipse.debug.core.ILaunchConfigurationWorkingCopy; -import org.eclipse.debug.core.ILaunchManager; -import org.eclipse.debug.ui.CommonTab; -import org.eclipse.debug.ui.DebugUITools; -import org.eclipse.debug.ui.IDebugUIConstants; import org.eclipse.debug.ui.ILaunchShortcut2; -import org.eclipse.jdt.launching.IJavaLaunchConfigurationConstants; -import org.eclipse.jface.dialogs.ErrorDialog; -import org.eclipse.jface.viewers.ILabelProvider; import org.eclipse.jface.viewers.ISelection; import org.eclipse.jface.viewers.IStructuredSelection; -import org.eclipse.jface.window.Window; -import org.eclipse.swt.widgets.Display; import org.eclipse.ui.IEditorInput; import org.eclipse.ui.IEditorPart; -import org.eclipse.ui.IWorkbenchPage; -import org.eclipse.ui.dialogs.ElementListSelectionDialog; import org.eclipse.ui.editors.text.ILocationProvider; -import org.eclipse.ui.externaltools.internal.launchConfigurations.ExternalToolsUtil; /** * This class provides the Run/Debug As -> Ant Build launch shortcut. * */ -public class AntLaunchShortcut implements ILaunchShortcut2 { - - private boolean fShowDialog = false; - private static final int MAX_TARGET_APPEND_LENGTH = 30; - private static final String DEFAULT_TARGET = "default"; //$NON-NLS-1$ - - @Override - public void launch(ISelection selection, String mode) { - if (selection instanceof IStructuredSelection) { - IStructuredSelection structuredSelection = (IStructuredSelection) selection; - Object object = structuredSelection.getFirstElement(); - if (object instanceof IAdaptable) { - if (object instanceof AntElementNode) { - launch((AntElementNode) object, mode); - return; - } - IResource resource = ((IAdaptable) object).getAdapter(IResource.class); - if (resource != null) { - if (!(AntUtil.isKnownAntFile(resource))) { - if (!AntUtil.isKnownBuildfileName(resource.getName())) { - if (resource.getType() == IResource.FILE) { - resource = resource.getParent(); - } - resource = findBuildFile((IContainer) resource); - } - } - if (resource != null) { - IFile file = (IFile) resource; - launch(file.getFullPath(), file.getProject(), mode, null); - return; - } - } - } - } - antFileNotFound(); - } - - /** - * Launches the given Ant node. - * - * - * @param node - * the Ant node to use as the context for the launch - * @param mode - * the mode of the launch - */ - public void launch(AntElementNode node, String mode) { - String selectedTargetName = null; - if (node instanceof AntTargetNode) { - AntTargetNode targetNode = (AntTargetNode) node; - if (targetNode.isDefaultTarget()) { - selectedTargetName = DEFAULT_TARGET; - } else { - // append a comma to be consistent with ant targets tab - selectedTargetName = targetNode.getTarget().getName() + ','; - } - } else if (node instanceof AntProjectNode) { - selectedTargetName = DEFAULT_TARGET; - } else if (node instanceof AntTaskNode) { - AntTaskNode taskNode = (AntTaskNode) node; - selectedTargetName = taskNode.getTask().getOwningTarget().getName(); - } - - IFile file = node.getBuildFileResource(); - if (file != null) { - launch(file.getFullPath(), file.getProject(), mode, selectedTargetName); - return; - } - // external buildfile - IWorkbenchPage page = AntUIPlugin.getActiveWorkbenchWindow().getActivePage(); - IPath filePath = null; - IEditorPart editor = page.getActiveEditor(); - if (editor != null) { - IEditorInput editorInput = editor.getEditorInput(); - ILocationProvider locationProvider = editorInput.getAdapter(ILocationProvider.class); - if (locationProvider != null) { - filePath = locationProvider.getPath(editorInput); - if (filePath != null) { - launch(filePath, null, mode, selectedTargetName); - return; - } - } - } - antFileNotFound(); - } - - /** - * Inform the user that an ant file was not found to run. - */ - private void antFileNotFound() { - reportError(AntLaunchConfigurationMessages.AntLaunchShortcut_Unable, null); - } - - /** - * Walks the file hierarchy looking for a build file. Returns the first build file found that matches the search criteria. - */ - private IFile findBuildFile(IContainer parent) { - String[] names = AntUtil.getKnownBuildfileNames(); - if (names == null) { - return null; - } - IContainer lparent = parent; - IResource file = null; - while (file == null || file.getType() != IResource.FILE) { - for (String name : names) { - file = lparent.findMember(name); - if (file != null && file.getType() == IResource.FILE) { - break; - } - } - lparent = lparent.getParent(); - if (lparent == null) { - return null; - } - } - return (IFile) file; - } - - /** - * Returns a listing of ILaunchConfigurations that correspond to the specified build file. - * - * @param filepath - * the path to the buildfile to launch - * @return the list of ILaunchConfigurations that correspond to the specified build file. - * - * @since 3.4 - */ - protected List collectConfigurations(IPath filepath) { - ILaunchManager manager = DebugPlugin.getDefault().getLaunchManager(); - ILaunchConfigurationType type = manager.getLaunchConfigurationType(IAntLaunchConstants.ID_ANT_LAUNCH_CONFIGURATION_TYPE); - if (type != null) { - try { - ILaunchConfiguration[] configs = manager.getLaunchConfigurations(type); - ArrayList list = new ArrayList<>(); - IPath location = null; - for (ILaunchConfiguration config : configs) { - if (config.exists()) { - try { - location = ExternalToolsUtil.getLocation(config); - if (location != null && location.equals(filepath)) { - list.add(config); - } - } - catch (CoreException ce) { - // do nothing - } - } - } - return list; - } - catch (CoreException e) { - // do nothing - } - } - return Collections.EMPTY_LIST; - } - - /** - * Returns a unique name for a copy of the given launch configuration with the given targets. The name seed is the same as the name for a new - * launch configuration with " [targetList]" appended to the end. - * - * @param filePath - * the path to the buildfile - * @param projectName - * the name of the project containing the buildfile or null if no project is known - * @param targetAttribute - * the listing of targets to execute or null for default target execution - * @return a unique name for the copy - */ - public static String getNewLaunchConfigurationName(IPath filePath, String projectName, String targetAttribute) { - StringBuilder buffer = new StringBuilder(); - if (projectName != null) { - buffer.append(projectName); - buffer.append(' '); - buffer.append(filePath.lastSegment()); - } else { - buffer.append(filePath.lastSegment()); - } - - if (targetAttribute != null) { - buffer.append(" ["); //$NON-NLS-1$ - if (targetAttribute.length() > MAX_TARGET_APPEND_LENGTH + 3) { - // The target attribute can potentially be a long, comma-separated list - // of target. Make sure the generated name isn't extremely long. - buffer.append(targetAttribute.substring(0, MAX_TARGET_APPEND_LENGTH)); - buffer.append("..."); //$NON-NLS-1$ - } else { - buffer.append(targetAttribute); - } - buffer.append(']'); - } - - String name = DebugPlugin.getDefault().getLaunchManager().generateLaunchConfigurationName(buffer.toString()); - return name; - } - - /** - * Launch the given targets in the given build file. The targets are launched in the given mode. - * - * @param filePath - * the path to the build file to launch - * @param project - * the project for the path - * @param mode - * the mode in which the build file should be executed - * @param targetAttribute - * the targets to launch or null to use targets on existing configuration, or DEFAULT for default target - * explicitly. - * - * configuration targets attribute. - */ - public void launch(IPath filePath, IProject project, String mode, String targetAttribute) { - ILaunchConfiguration configuration = null; - IFile backingfile = null; - if (project != null) { - // need to get the full location of a workspace file to compare against the resolved config location attribute - backingfile = project.getFile(filePath.removeFirstSegments(1)); - } - List configs = collectConfigurations((backingfile != null && backingfile.exists() ? backingfile.getLocation() - : filePath)); - if (configs.isEmpty()) { - configuration = createDefaultLaunchConfiguration(filePath, (project != null && project.exists() ? project : null)); - } else if (configs.size() == 1) { - configuration = configs.get(0); - } else { - configuration = chooseConfig(configs); - if (configuration == null) { - // fail gracefully if the user cancels choosing a configuration - return; - } - } - - // set the target to run, if applicable - if (configuration != null) { - try { - if (targetAttribute != null - && !targetAttribute.equals(configuration.getAttribute(IAntLaunchConstants.ATTR_ANT_TARGETS, DEFAULT_TARGET))) { - ILaunchConfigurationWorkingCopy copy = configuration.getWorkingCopy(); - String attrValue = null; - if (!DEFAULT_TARGET.equals(targetAttribute)) { - attrValue = targetAttribute; - } - copy.setAttribute(IAntLaunchConstants.ATTR_ANT_TARGETS, attrValue); - configuration = copy.doSave(); - } - } - catch (CoreException exception) { - reportError(MessageFormat.format(AntLaunchConfigurationMessages.AntLaunchShortcut_Exception_launching, new Object[] { - filePath.toFile().getName() }), exception); - return; - } - launch(mode, configuration); - } else { - antFileNotFound(); - } - } - - /** - * Delegate method to launch the specified ILaunchConfiguration in the specified mode - * - * @param mode - * the mode to launch in - * @param configuration - * the ILaunchConfiguration to launch - */ - private void launch(String mode, ILaunchConfiguration configuration) { - if (fShowDialog) { - /* - * // Offer to save dirty editors before opening the dialog as the contents // of an Ant editor often affect the contents of the dialog. - * if (!DebugUITools.saveBeforeLaunch()) { return; } - */ - IStatus status = new Status(IStatus.INFO, IAntUIConstants.PLUGIN_ID, IAntUIConstants.STATUS_INIT_RUN_ANT, IAntCoreConstants.EMPTY_STRING, null); - String groupId; - if (mode.equals(ILaunchManager.DEBUG_MODE)) { - groupId = IDebugUIConstants.ID_DEBUG_LAUNCH_GROUP; - } else { - groupId = org.eclipse.ui.externaltools.internal.model.IExternalToolConstants.ID_EXTERNAL_TOOLS_LAUNCH_GROUP; - } - DebugUITools.openLaunchConfigurationDialog(AntUIPlugin.getActiveWorkbenchWindow().getShell(), configuration, groupId, status); - } else { - DebugUITools.launch(configuration, mode); - } - } - - /** - * Creates and returns a default launch configuration for the given file. - * - * @param file - * @return default launch configuration - */ - public static ILaunchConfiguration createDefaultLaunchConfiguration(IFile file) { - return createDefaultLaunchConfiguration(file.getFullPath(), file.getProject()); - } - - /** - * Creates and returns a default launch configuration for the given file path and project. - * - * @param filePath - * the path to the buildfile - * @param project - * the project containing the buildfile or null if the buildfile is not contained in a project (is external). - * @return default launch configuration or null if one could not be created - */ - public static ILaunchConfiguration createDefaultLaunchConfiguration(IPath filePath, IProject project) { - ILaunchManager manager = DebugPlugin.getDefault().getLaunchManager(); - ILaunchConfigurationType type = manager.getLaunchConfigurationType(IAntLaunchConstants.ID_ANT_LAUNCH_CONFIGURATION_TYPE); - - String projectName = project != null ? project.getName() : null; - String name = getNewLaunchConfigurationName(filePath, projectName, null); - try { - ILaunchConfigurationWorkingCopy workingCopy = type.newInstance(null, name); - if (project != null) { - workingCopy.setAttribute(IExternalToolConstants.ATTR_LOCATION, VariablesPlugin.getDefault().getStringVariableManager().generateVariableExpression("workspace_loc", filePath.toString())); //$NON-NLS-1$ - } else { - workingCopy.setAttribute(IExternalToolConstants.ATTR_LOCATION, filePath.toString()); - } - workingCopy.setAttribute(IJavaLaunchConfigurationConstants.ATTR_CLASSPATH_PROVIDER, "org.eclipse.ant.ui.AntClasspathProvider"); //$NON-NLS-1$ - // set default for common settings - CommonTab tab = new CommonTab(); - tab.setDefaults(workingCopy); - tab.dispose(); - - // set the project name so that the correct default VM install can be determined - if (project != null) { - workingCopy.setAttribute(IJavaLaunchConfigurationConstants.ATTR_PROJECT_NAME, project.getName()); - } - AntJRETab jreTab = new AntJRETab(); - jreTab.setDefaults(workingCopy); - jreTab.dispose(); - - IFile file = AntUtil.getFileForLocation(filePath.toString(), null); - workingCopy.setMappedResources(new IResource[] { file }); - - return workingCopy.doSave(); - } - catch (CoreException e) { - reportError(MessageFormat.format(AntLaunchConfigurationMessages.AntLaunchShortcut_2, new Object[] { filePath.toString() }), e); - } - return null; - } - - /** - * Returns a list of existing launch configuration for the given file. - * - * @param file - * the buildfile resource - * @return list of launch configurations - */ - public static List findExistingLaunchConfigurations(IFile file) { - List validConfigs = new ArrayList<>(); - if (file != null) { - IPath filePath = file.getLocation(); - if (filePath != null) { - ILaunchManager manager = DebugPlugin.getDefault().getLaunchManager(); - ILaunchConfigurationType type = manager.getLaunchConfigurationType(IAntLaunchConstants.ID_ANT_LAUNCH_CONFIGURATION_TYPE); - if (type != null) { - try { - for (ILaunchConfiguration config : manager.getLaunchConfigurations(type)) { - try { - if (filePath.equals(ExternalToolsUtil.getLocation(config))) { - validConfigs.add(config); - } - } - catch (CoreException ce) { - // do nothing - } - } - } - catch (CoreException e) { - reportError(AntLaunchConfigurationMessages.AntLaunchShortcut_3, e); - } - } - } - } - return validConfigs; - } - - /** - * Prompts the user to choose from the list of given launch configurations and returns the config the user choose or null if the user - * pressed Cancel or if the given list is empty. - * - * @param configs - * the list of {@link ILaunchConfiguration}s to choose from - * @return the chosen {@link ILaunchConfiguration} or null - */ - public static ILaunchConfiguration chooseConfig(List configs) { - if (configs.isEmpty()) { - return null; - } - ILabelProvider labelProvider = DebugUITools.newDebugModelPresentation(); - ElementListSelectionDialog dialog = new ElementListSelectionDialog(Display.getDefault().getActiveShell(), labelProvider); - dialog.setElements(configs.toArray(new ILaunchConfiguration[configs.size()])); - dialog.setTitle(AntLaunchConfigurationMessages.AntLaunchShortcut_4); - dialog.setMessage(AntLaunchConfigurationMessages.AntLaunchShortcut_5); - dialog.setMultipleSelection(false); - int result = dialog.open(); - labelProvider.dispose(); - if (result == Window.OK) { - return (ILaunchConfiguration) dialog.getFirstResult(); - } - return null; - } - - @Override - public void launch(IEditorPart editor, String mode) { - IEditorInput input = editor.getEditorInput(); - IFile file = input.getAdapter(IFile.class); - IPath filepath = null; - if (file != null) { - filepath = file.getFullPath(); - } - if (filepath == null) { - ILocationProvider locationProvider = input.getAdapter(ILocationProvider.class); - if (locationProvider != null) { - filepath = locationProvider.getPath(input); - } - } - if (filepath != null && (AntUtil.isKnownAntFile(file) || AntUtil.isKnownAntFile(filepath.toFile()))) { - launch(filepath, (file == null ? null : file.getProject()), mode, null); - return; - } - if (file != null) { - if (!AntUtil.isKnownBuildfileName(file.getName())) { - file = findBuildFile(file.getParent()); - } - if (file != null) { - launch(file.getFullPath(), file.getProject(), mode, null); - return; - } - } - antFileNotFound(); - } - - /** - * Opens an error dialog presenting the user with the specified message and throwable - * - * @param message - * @param throwable - */ - protected static void reportError(String message, Throwable throwable) { - IStatus status = null; - if (throwable instanceof CoreException) { - status = ((CoreException) throwable).getStatus(); - } else { - status = new Status(IStatus.ERROR, IAntUIConstants.PLUGIN_ID, 0, message, throwable); - } - ErrorDialog.openError(AntUIPlugin.getActiveWorkbenchWindow().getShell(), AntLaunchConfigurationMessages.AntLaunchShortcut_Error_7, AntLaunchConfigurationMessages.AntLaunchShortcut_Build_Failed_2, status); - } - - /** - * Sets whether to show the external tools launch configuration dialog - * - * @param showDialog - * If true the launch configuration dialog will always be shown - */ - public void setShowDialog(boolean showDialog) { - fShowDialog = showDialog; - } +public class AntLaunchShortcut extends AntLaunchShortcutCommon implements ILaunchShortcut2 { @Override public ILaunchConfiguration[] getLaunchConfigurations(ISelection selection) { diff --git a/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcutCommon.java b/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcutCommon.java new file mode 100644 index 00000000000..1559a8f9ea6 --- /dev/null +++ b/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcutCommon.java @@ -0,0 +1,530 @@ +/******************************************************************************* + * Copyright (c) ETAS GmbH 2023, all rights reserved. + * + * This program and the accompanying materials + * are made available under the terms of the Eclipse Public License 2.0 + * which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + * + * Contributors: + * ETAS GmbH - Moved the common launch APIs here from AntLaunchShortcut + * + *******************************************************************************/ + +package org.eclipse.ant.internal.ui.launchConfigurations; + +import java.text.MessageFormat; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; + +import org.eclipse.ant.internal.core.IAntCoreConstants; +import org.eclipse.ant.internal.ui.AntUIPlugin; +import org.eclipse.ant.internal.ui.AntUtil; +import org.eclipse.ant.internal.ui.IAntUIConstants; +import org.eclipse.ant.internal.ui.model.AntElementNode; +import org.eclipse.ant.internal.ui.model.AntProjectNode; +import org.eclipse.ant.internal.ui.model.AntTargetNode; +import org.eclipse.ant.internal.ui.model.AntTaskNode; +import org.eclipse.ant.launching.IAntLaunchConstants; +import org.eclipse.core.externaltools.internal.IExternalToolConstants; +import org.eclipse.core.resources.IContainer; +import org.eclipse.core.resources.IFile; +import org.eclipse.core.resources.IProject; +import org.eclipse.core.resources.IResource; +import org.eclipse.core.runtime.CoreException; +import org.eclipse.core.runtime.IAdaptable; +import org.eclipse.core.runtime.IPath; +import org.eclipse.core.runtime.IStatus; +import org.eclipse.core.runtime.Status; +import org.eclipse.core.variables.VariablesPlugin; +import org.eclipse.debug.core.DebugPlugin; +import org.eclipse.debug.core.ILaunchConfiguration; +import org.eclipse.debug.core.ILaunchConfigurationType; +import org.eclipse.debug.core.ILaunchConfigurationWorkingCopy; +import org.eclipse.debug.core.ILaunchManager; +import org.eclipse.debug.ui.CommonTab; +import org.eclipse.debug.ui.DebugUITools; +import org.eclipse.debug.ui.IDebugUIConstants; +import org.eclipse.debug.ui.ILaunchShortcut; +import org.eclipse.jdt.launching.IJavaLaunchConfigurationConstants; +import org.eclipse.jface.dialogs.ErrorDialog; +import org.eclipse.jface.viewers.ILabelProvider; +import org.eclipse.jface.viewers.ISelection; +import org.eclipse.jface.viewers.IStructuredSelection; +import org.eclipse.jface.window.Window; +import org.eclipse.swt.widgets.Display; +import org.eclipse.ui.IEditorInput; +import org.eclipse.ui.IEditorPart; +import org.eclipse.ui.IWorkbenchPage; +import org.eclipse.ui.dialogs.ElementListSelectionDialog; +import org.eclipse.ui.editors.text.ILocationProvider; +import org.eclipse.ui.externaltools.internal.launchConfigurations.ExternalToolsUtil; + +public class AntLaunchShortcutCommon implements ILaunchShortcut { + private boolean fShowDialog = false; + private static final int MAX_TARGET_APPEND_LENGTH = 30; + private static final String DEFAULT_TARGET = "default"; //$NON-NLS-1$ + + @Override + public void launch(ISelection selection, String mode) { + if (selection instanceof IStructuredSelection) { + IStructuredSelection structuredSelection = (IStructuredSelection) selection; + Object object = structuredSelection.getFirstElement(); + if (object instanceof IAdaptable) { + if (object instanceof AntElementNode) { + launch((AntElementNode) object, mode); + return; + } + IResource resource = ((IAdaptable) object).getAdapter(IResource.class); + if (resource != null) { + if (!(AntUtil.isKnownAntFile(resource))) { + if (!AntUtil.isKnownBuildfileName(resource.getName())) { + if (resource.getType() == IResource.FILE) { + resource = resource.getParent(); + } + resource = findBuildFile((IContainer) resource); + } + } + if (resource != null) { + IFile file = (IFile) resource; + launch(file.getFullPath(), file.getProject(), mode, null); + return; + } + } + } + } + antFileNotFound(); + + } + + /** + * Launch the given targets in the given build file. The targets are launched in the given mode. + * + * @param filePath + * the path to the build file to launch + * @param project + * the project for the path + * @param mode + * the mode in which the build file should be executed + * @param targetAttribute + * the targets to launch or null to use targets on existing configuration, or DEFAULT for default target + * explicitly. + * + * configuration targets attribute. + */ + public void launch(IPath filePath, IProject project, String mode, String targetAttribute) { + ILaunchConfiguration configuration = null; + IFile backingfile = null; + if (project != null) { + // need to get the full location of a workspace file to compare against the resolved config location attribute + backingfile = project.getFile(filePath.removeFirstSegments(1)); + } + List configs = collectConfigurations((backingfile != null && backingfile.exists() ? backingfile.getLocation() + : filePath)); + if (configs.isEmpty()) { + configuration = createDefaultLaunchConfiguration(filePath, (project != null && project.exists() ? project : null)); + } else if (configs.size() == 1) { + configuration = configs.get(0); + } else { + configuration = chooseConfig(configs); + if (configuration == null) { + // fail gracefully if the user cancels choosing a configuration + return; + } + } + + // set the target to run, if applicable + if (configuration != null) { + try { + if (targetAttribute != null + && !targetAttribute.equals(configuration.getAttribute(IAntLaunchConstants.ATTR_ANT_TARGETS, DEFAULT_TARGET))) { + ILaunchConfigurationWorkingCopy copy = configuration.getWorkingCopy(); + String attrValue = null; + if (!DEFAULT_TARGET.equals(targetAttribute)) { + attrValue = targetAttribute; + } + copy.setAttribute(IAntLaunchConstants.ATTR_ANT_TARGETS, attrValue); + configuration = copy.doSave(); + } + } + catch (CoreException exception) { + reportError(MessageFormat.format(AntLaunchConfigurationMessages.AntLaunchShortcut_Exception_launching, new Object[] { + filePath.toFile().getName() }), exception); + return; + } + launch(mode, configuration); + } else { + antFileNotFound(); + } + } + + /** + * Delegate method to launch the specified ILaunchConfiguration in the specified mode + * + * @param mode + * the mode to launch in + * @param configuration + * the ILaunchConfiguration to launch + */ + private void launch(String mode, ILaunchConfiguration configuration) { + if (fShowDialog) { + /* + * // Offer to save dirty editors before opening the dialog as the contents // of an Ant editor often affect the contents of the dialog. + * if (!DebugUITools.saveBeforeLaunch()) { return; } + */ + IStatus status = new Status(IStatus.INFO, IAntUIConstants.PLUGIN_ID, IAntUIConstants.STATUS_INIT_RUN_ANT, IAntCoreConstants.EMPTY_STRING, null); + String groupId; + if (mode.equals(ILaunchManager.DEBUG_MODE)) { + groupId = IDebugUIConstants.ID_DEBUG_LAUNCH_GROUP; + } else { + groupId = org.eclipse.ui.externaltools.internal.model.IExternalToolConstants.ID_EXTERNAL_TOOLS_LAUNCH_GROUP; + } + DebugUITools.openLaunchConfigurationDialog(AntUIPlugin.getActiveWorkbenchWindow().getShell(), configuration, groupId, status); + } else { + DebugUITools.launch(configuration, mode); + } + } + + /** + * Launches the given Ant node. + *
    + *
  • AntProjectNodes: the default target is executed
  • + *
  • AntTargetNodes: that target is executed
  • + *
  • AntTaskNodes: the owning target is executed
  • + *
+ * + * @param node + * the Ant node to use as the context for the launch + * @param mode + * the mode of the launch + */ + public void launch(AntElementNode node, String mode) { + String selectedTargetName = null; + if (node instanceof AntTargetNode) { + AntTargetNode targetNode = (AntTargetNode) node; + if (targetNode.isDefaultTarget()) { + selectedTargetName = DEFAULT_TARGET; + } else { + // append a comma to be consistent with ant targets tab + selectedTargetName = targetNode.getTarget().getName() + ','; + } + } else if (node instanceof AntProjectNode) { + selectedTargetName = DEFAULT_TARGET; + } else if (node instanceof AntTaskNode) { + AntTaskNode taskNode = (AntTaskNode) node; + selectedTargetName = taskNode.getTask().getOwningTarget().getName(); + } + + IFile file = node.getBuildFileResource(); + if (file != null) { + launch(file.getFullPath(), file.getProject(), mode, selectedTargetName); + return; + } + // external buildfile + IWorkbenchPage page = AntUIPlugin.getActiveWorkbenchWindow().getActivePage(); + IPath filePath = null; + IEditorPart editor = page.getActiveEditor(); + if (editor != null) { + IEditorInput editorInput = editor.getEditorInput(); + ILocationProvider locationProvider = editorInput.getAdapter(ILocationProvider.class); + if (locationProvider != null) { + filePath = locationProvider.getPath(editorInput); + if (filePath != null) { + launch(filePath, null, mode, selectedTargetName); + return; + } + } + } + antFileNotFound(); + } + + /** + * Walks the file hierarchy looking for a build file. Returns the first build file found that matches the search criteria. + */ + protected IFile findBuildFile(IContainer parent) { + String[] names = AntUtil.getKnownBuildfileNames(); + if (names == null) { + return null; + } + IContainer lparent = parent; + IResource file = null; + while (file == null || file.getType() != IResource.FILE) { + for (String name : names) { + file = lparent.findMember(name); + if (file != null && file.getType() == IResource.FILE) { + break; + } + } + lparent = lparent.getParent(); + if (lparent == null) { + return null; + } + } + return (IFile) file; + } + + /** + * Inform the user that an ant file was not found to run. + */ + protected void antFileNotFound() { + reportError(AntLaunchConfigurationMessages.AntLaunchShortcut_Unable, null); + } + + /** + * Opens an error dialog presenting the user with the specified message and throwable + * + * @param message + * @param throwable + */ + protected static void reportError(String message, Throwable throwable) { + IStatus status = null; + if (throwable instanceof CoreException) { + status = ((CoreException) throwable).getStatus(); + } else { + status = new Status(IStatus.ERROR, IAntUIConstants.PLUGIN_ID, 0, message, throwable); + } + ErrorDialog.openError(AntUIPlugin.getActiveWorkbenchWindow().getShell(), AntLaunchConfigurationMessages.AntLaunchShortcut_Error_7, AntLaunchConfigurationMessages.AntLaunchShortcut_Build_Failed_2, status); + } + + /** + * Returns a listing of ILaunchConfigurations that correspond to the specified build file. + * + * @param filepath + * the path to the buildfile to launch + * @return the list of ILaunchConfigurations that correspond to the specified build file. + * + * @since 3.4 + */ + protected List collectConfigurations(IPath filepath) { + ILaunchManager manager = DebugPlugin.getDefault().getLaunchManager(); + ILaunchConfigurationType type = manager.getLaunchConfigurationType(IAntLaunchConstants.ID_ANT_LAUNCH_CONFIGURATION_TYPE); + if (type != null) { + try { + ILaunchConfiguration[] configs = manager.getLaunchConfigurations(type); + ArrayList list = new ArrayList<>(); + IPath location = null; + for (ILaunchConfiguration config : configs) { + if (config.exists()) { + try { + location = ExternalToolsUtil.getLocation(config); + if (location != null && location.equals(filepath)) { + list.add(config); + } + } + catch (CoreException ce) { + // do nothing + } + } + } + return list; + } + catch (CoreException e) { + // do nothing + } + } + return Collections.EMPTY_LIST; + } + + /** + * Creates and returns a default launch configuration for the given file. + * + * @param file + * @return default launch configuration + */ + public static ILaunchConfiguration createDefaultLaunchConfiguration(IFile file) { + return createDefaultLaunchConfiguration(file.getFullPath(), file.getProject()); + } + + /** + * Creates and returns a default launch configuration for the given file path and project. + * + * @param filePath + * the path to the buildfile + * @param project + * the project containing the buildfile or null if the buildfile is not contained in a project (is external). + * @return default launch configuration or null if one could not be created + */ + public static ILaunchConfiguration createDefaultLaunchConfiguration(IPath filePath, IProject project) { + ILaunchManager manager = DebugPlugin.getDefault().getLaunchManager(); + ILaunchConfigurationType type = manager.getLaunchConfigurationType(IAntLaunchConstants.ID_ANT_LAUNCH_CONFIGURATION_TYPE); + + String projectName = project != null ? project.getName() : null; + String name = getNewLaunchConfigurationName(filePath, projectName, null); + try { + ILaunchConfigurationWorkingCopy workingCopy = type.newInstance(null, name); + if (project != null) { + workingCopy.setAttribute(IExternalToolConstants.ATTR_LOCATION, VariablesPlugin.getDefault().getStringVariableManager().generateVariableExpression("workspace_loc", filePath.toString())); //$NON-NLS-1$ + } else { + workingCopy.setAttribute(IExternalToolConstants.ATTR_LOCATION, filePath.toString()); + } + workingCopy.setAttribute(IJavaLaunchConfigurationConstants.ATTR_CLASSPATH_PROVIDER, "org.eclipse.ant.ui.AntClasspathProvider"); //$NON-NLS-1$ + // set default for common settings + CommonTab tab = new CommonTab(); + tab.setDefaults(workingCopy); + tab.dispose(); + + // set the project name so that the correct default VM install can be determined + if (project != null) { + workingCopy.setAttribute(IJavaLaunchConfigurationConstants.ATTR_PROJECT_NAME, project.getName()); + } + AntJRETab jreTab = new AntJRETab(); + jreTab.setDefaults(workingCopy); + jreTab.dispose(); + + IFile file = AntUtil.getFileForLocation(filePath.toString(), null); + workingCopy.setMappedResources(new IResource[] { file }); + + return workingCopy.doSave(); + } + catch (CoreException e) { + reportError(MessageFormat.format(AntLaunchConfigurationMessages.AntLaunchShortcut_2, new Object[] { filePath.toString() }), e); + } + return null; + } + + /** + * Returns a unique name for a copy of the given launch configuration with the given targets. The name seed is the same as the name for a new + * launch configuration with " [targetList]" appended to the end. + * + * @param filePath + * the path to the buildfile + * @param projectName + * the name of the project containing the buildfile or null if no project is known + * @param targetAttribute + * the listing of targets to execute or null for default target execution + * @return a unique name for the copy + */ + public static String getNewLaunchConfigurationName(IPath filePath, String projectName, String targetAttribute) { + StringBuilder buffer = new StringBuilder(); + if (projectName != null) { + buffer.append(projectName); + buffer.append(' '); + buffer.append(filePath.lastSegment()); + } else { + buffer.append(filePath.lastSegment()); + } + + if (targetAttribute != null) { + buffer.append(" ["); //$NON-NLS-1$ + if (targetAttribute.length() > MAX_TARGET_APPEND_LENGTH + 3) { + // The target attribute can potentially be a long, comma-separated list + // of target. Make sure the generated name isn't extremely long. + buffer.append(targetAttribute.substring(0, MAX_TARGET_APPEND_LENGTH)); + buffer.append("..."); //$NON-NLS-1$ + } else { + buffer.append(targetAttribute); + } + buffer.append(']'); + } + + String name = DebugPlugin.getDefault().getLaunchManager().generateLaunchConfigurationName(buffer.toString()); + return name; + } + + /** + * Prompts the user to choose from the list of given launch configurations and returns the config the user choose or null if the user + * pressed Cancel or if the given list is empty. + * + * @param configs + * the list of {@link ILaunchConfiguration}s to choose from + * @return the chosen {@link ILaunchConfiguration} or null + */ + public static ILaunchConfiguration chooseConfig(List configs) { + if (configs.isEmpty()) { + return null; + } + ILabelProvider labelProvider = DebugUITools.newDebugModelPresentation(); + ElementListSelectionDialog dialog = new ElementListSelectionDialog(Display.getDefault().getActiveShell(), labelProvider); + dialog.setElements(configs.toArray(new ILaunchConfiguration[configs.size()])); + dialog.setTitle(AntLaunchConfigurationMessages.AntLaunchShortcut_4); + dialog.setMessage(AntLaunchConfigurationMessages.AntLaunchShortcut_5); + dialog.setMultipleSelection(false); + int result = dialog.open(); + labelProvider.dispose(); + if (result == Window.OK) { + return (ILaunchConfiguration) dialog.getFirstResult(); + } + return null; + } + + @Override + public void launch(IEditorPart editor, String mode) { + IEditorInput input = editor.getEditorInput(); + IFile file = input.getAdapter(IFile.class); + IPath filepath = null; + if (file != null) { + filepath = file.getFullPath(); + } + if (filepath == null) { + ILocationProvider locationProvider = input.getAdapter(ILocationProvider.class); + if (locationProvider != null) { + filepath = locationProvider.getPath(input); + } + } + if (filepath != null && (AntUtil.isKnownAntFile(file) || AntUtil.isKnownAntFile(filepath.toFile()))) { + launch(filepath, (file == null ? null : file.getProject()), mode, null); + return; + } + if (file != null) { + if (!AntUtil.isKnownBuildfileName(file.getName())) { + file = findBuildFile(file.getParent()); + } + if (file != null) { + launch(file.getFullPath(), file.getProject(), mode, null); + return; + } + } + antFileNotFound(); + + } + + /** + * Sets whether to show the external tools launch configuration dialog + * + * @param showDialog + * If true the launch configuration dialog will always be shown + */ + public void setShowDialog(boolean showDialog) { + fShowDialog = showDialog; + } + + /** + * Returns a list of existing launch configuration for the given file. + * + * @param file + * the buildfile resource + * @return list of launch configurations + */ + public static List findExistingLaunchConfigurations(IFile file) { + List validConfigs = new ArrayList<>(); + if (file != null) { + IPath filePath = file.getLocation(); + if (filePath != null) { + ILaunchManager manager = DebugPlugin.getDefault().getLaunchManager(); + ILaunchConfigurationType type = manager.getLaunchConfigurationType(IAntLaunchConstants.ID_ANT_LAUNCH_CONFIGURATION_TYPE); + if (type != null) { + try { + for (ILaunchConfiguration config : manager.getLaunchConfigurations(type)) { + try { + if (filePath.equals(ExternalToolsUtil.getLocation(config))) { + validConfigs.add(config); + } + } + catch (CoreException ce) { + // do nothing + } + } + } + catch (CoreException e) { + reportError(AntLaunchConfigurationMessages.AntLaunchShortcut_3, e); + } + } + } + } + return validConfigs; + } + +} diff --git a/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcutWithDialog.java b/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcutWithDialog.java index 2a80d36c4a6..295af407e76 100644 --- a/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcutWithDialog.java +++ b/ant/org.eclipse.ant.ui/Ant Tools Support/org/eclipse/ant/internal/ui/launchConfigurations/AntLaunchShortcutWithDialog.java @@ -10,10 +10,11 @@ * * Contributors: * IBM Corporation - initial API and implementation + * Sudhindra Kulkarni (ETAS GmbH) - Issue 527 *******************************************************************************/ package org.eclipse.ant.internal.ui.launchConfigurations; -public class AntLaunchShortcutWithDialog extends AntLaunchShortcut { +public class AntLaunchShortcutWithDialog extends AntLaunchShortcutCommon { /** * Creates a new Ant launch shortcut that will open the launch configuration dialog.