blob: f0480162e6b5baf48a6df65bc4ff3251bbeabf72 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2015 IBM Corporation and others.
*
* 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:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.ui.internal;
import java.util.ArrayList;
import java.util.Iterator;
import org.eclipse.core.runtime.IConfigurationElement;
import org.eclipse.jface.action.IAction;
import org.eclipse.jface.viewers.ISelection;
import org.eclipse.swt.events.HelpListener;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.ui.IActionDelegate;
import org.eclipse.ui.IWorkbenchPart;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.IWorkbenchWindowActionDelegate;
import org.eclipse.ui.WorkbenchException;
import org.eclipse.ui.actions.LabelRetargetAction;
import org.eclipse.ui.actions.RetargetAction;
import org.eclipse.ui.internal.registry.IWorkbenchRegistryConstants;
/**
* This class extends regular plugin action with the additional requirement that
* the delegate has to implement interface
* {@link org.eclipse.ui.IWorkbenchWindowActionDelegate}. This interface has one
* additional method (init) whose purpose is to initialize the delegate with the
* window in which the action is intended to run.
*/
public class WWinPluginAction extends PluginAction implements IActionSetContributionItem {
/**
* The help listener assigned to this action, or <code>null</code> if none.
*/
private HelpListener localHelpListener;
private IWorkbenchWindow window;
private String actionSetId;
private RetargetAction retargetAction;
private static ArrayList<WWinPluginAction> staticActionList = new ArrayList<>(50);
/**
* Constructs a new <code>WWinPluginAction</code> object.
*
* @param actionElement the configuration element
* @param window the window to contribute to
* @param id the identifier
* @param style the style
*/
public WWinPluginAction(IConfigurationElement actionElement, IWorkbenchWindow window, String id, int style) {
super(actionElement, id, style);
this.window = window;
// If config specifies a retarget action, create it now
String retarget = actionElement.getAttribute(IWorkbenchRegistryConstants.ATT_RETARGET);
if (retarget != null && Boolean.parseBoolean(retarget)) {
// create a retarget action
String allowLabelUpdate = actionElement.getAttribute(IWorkbenchRegistryConstants.ATT_ALLOW_LABEL_UPDATE);
String label = actionElement.getAttribute(IWorkbenchRegistryConstants.ATT_LABEL);
if (allowLabelUpdate != null && Boolean.parseBoolean(allowLabelUpdate)) {
retargetAction = new LabelRetargetAction(id, label, style);
} else {
retargetAction = new RetargetAction(id, label, style);
}
retargetAction.addPropertyChangeListener(event -> {
if (event.getProperty().equals(IAction.ENABLED)) {
Object val1 = event.getNewValue();
if (val1 instanceof Boolean) {
setEnabled(((Boolean) val1).booleanValue());
}
} else if (event.getProperty().equals(IAction.CHECKED)) {
Object val2 = event.getNewValue();
if (val2 instanceof Boolean) {
setChecked(((Boolean) val2).booleanValue());
}
} else if (event.getProperty().equals(IAction.TEXT)) {
Object val3 = event.getNewValue();
if (val3 instanceof String) {
setText((String) val3);
}
} else if (event.getProperty().equals(IAction.TOOL_TIP_TEXT)) {
Object val4 = event.getNewValue();
if (val4 instanceof String) {
setToolTipText((String) val4);
}
}
});
retargetAction.setEnabled(false);
setEnabled(false);
window.getPartService().addPartListener(retargetAction);
IWorkbenchPart activePart = window.getPartService().getActivePart();
if (activePart != null) {
retargetAction.partActivated(activePart);
}
} else {
// if we retarget the handler will look after selection changes
window.getSelectionService().addSelectionListener(this);
refreshSelection();
}
addToActionList(this);
super.setHelpListener(e -> {
HelpListener listener = null;
if (retargetAction != null) {
listener = retargetAction.getHelpListener();
}
if (listener == null) {
// use our own help listener
listener = localHelpListener;
}
if (listener != null) {
// pass on the event
listener.helpRequested(e);
}
});
}
/**
* Adds an item to the action list.
*/
private static void addToActionList(WWinPluginAction action) {
staticActionList.add(action);
Shell shell = action.window.getShell();
if (shell != null) {
shell.addDisposeListener(x -> action.dispose());
}
}
/**
* Removes an item from the action list.
*/
private static void removeFromActionList(WWinPluginAction action) {
staticActionList.remove(action);
}
/**
* Creates any actions which belong to an activated plugin.
*/
public static void refreshActionList() {
Iterator<WWinPluginAction> iter = staticActionList.iterator();
while (iter.hasNext()) {
WWinPluginAction action = iter.next();
if ((action.getDelegate() == null) && action.isOkToCreateDelegate()) {
action.createDelegate();
// creating the delegate also refreshes its enablement
}
}
}
@Override
protected IActionDelegate validateDelegate(Object obj) throws WorkbenchException {
if (obj instanceof IWorkbenchWindowActionDelegate) {
return (IWorkbenchWindowActionDelegate) obj;
}
throw new WorkbenchException("Action must implement IWorkbenchWindowActionDelegate"); //$NON-NLS-1$
}
@Override
protected void initDelegate() {
super.initDelegate();
((IWorkbenchWindowActionDelegate) getDelegate()).init(window);
}
/**
* Disposes of the action and any resources held.
*/
@Override
public void dispose() {
if (window == null) {
return;
}
removeFromActionList(this);
if (retargetAction != null) {
window.getPartService().removePartListener(retargetAction);
retargetAction.dispose();
retargetAction = null;
}
window.getSelectionService().removeSelectionListener(this);
super.dispose();
window = null;
}
/**
* Returns the action set id.
*/
@Override
public String getActionSetId() {
return actionSetId;
}
/**
* Returns true if the window has been set. The window may be null after the
* constructor is called and before the window is stored. We cannot create the
* delegate at that time.
*/
@Override
public boolean isOkToCreateDelegate() {
return super.isOkToCreateDelegate() && window != null && retargetAction == null;
}
@Override
public void runWithEvent(Event event) {
if (retargetAction == null) {
super.runWithEvent(event);
return;
}
if (event != null) {
retargetAction.runWithEvent(event);
} else {
retargetAction.run();
}
}
/**
* Sets the action set id.
*/
@Override
public void setActionSetId(String newActionSetId) {
actionSetId = newActionSetId;
}
/**
* The <code>WWinPluginAction</code> implementation of this method declared on
* <code>IAction</code> stores the help listener in a local field. The supplied
* listener is only used if there is no retarget action.
*/
@Override
public void setHelpListener(HelpListener listener) {
localHelpListener = listener;
}
@Override
public void setChecked(boolean checked) {
super.setChecked(checked);
// This call may come from the SWT control event handler
// itself, so notify the retarget action to keep things
// in sync.
if (retargetAction != null) {
retargetAction.setChecked(checked);
}
}
/**
* Refresh the selection for the action.
*/
protected void refreshSelection() {
ISelection selection = window.getSelectionService().getSelection();
selectionChanged(selection);
}
@Override
public String toString() {
return "WWinPluginAction [" //$NON-NLS-1$
+ "id=" + getId() //$NON-NLS-1$
+ ", enabled=" + isEnabled() + //$NON-NLS-1$
(actionSetId != null ? ", actionSet=" + actionSetId : "") //$NON-NLS-1$ //$NON-NLS-2$
+ "]"; //$NON-NLS-1$
}
}