| /******************************************************************************* |
| * Copyright (c) 2000, 2007 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 |
| *******************************************************************************/ |
| package org.eclipse.ui.internal.dialogs; |
| |
| import java.lang.ref.SoftReference; |
| import java.util.ArrayList; |
| import java.util.Collection; |
| import java.util.HashMap; |
| import java.util.Iterator; |
| import java.util.Map; |
| |
| import org.eclipse.core.expressions.EvaluationContext; |
| import org.eclipse.core.expressions.EvaluationResult; |
| import org.eclipse.core.expressions.Expression; |
| import org.eclipse.core.expressions.ExpressionConverter; |
| import org.eclipse.core.runtime.CoreException; |
| import org.eclipse.core.runtime.IAdaptable; |
| import org.eclipse.core.runtime.IConfigurationElement; |
| import org.eclipse.core.runtime.IStatus; |
| import org.eclipse.core.runtime.Status; |
| import org.eclipse.jface.resource.ImageDescriptor; |
| import org.eclipse.ui.IActionFilter; |
| import org.eclipse.ui.IWorkbenchPropertyPage; |
| import org.eclipse.ui.SelectionEnabler; |
| import org.eclipse.ui.internal.LegacyResourceSupport; |
| import org.eclipse.ui.internal.WorkbenchPlugin; |
| import org.eclipse.ui.internal.registry.CategorizedPageRegistryReader; |
| import org.eclipse.ui.internal.registry.IWorkbenchRegistryConstants; |
| import org.eclipse.ui.internal.registry.PropertyPagesRegistryReader; |
| import org.eclipse.ui.internal.util.Util; |
| import org.eclipse.ui.model.IWorkbenchAdapter; |
| import org.eclipse.ui.plugin.AbstractUIPlugin; |
| |
| /** |
| * This property page contributor is created from page entry in the registry. |
| * Since instances of this class are created by the workbench, there is no |
| * danger of premature loading of plugins. |
| */ |
| |
| public class RegistryPageContributor implements IPropertyPageContributor, |
| IAdaptable { |
| private static final String CHILD_ENABLED_WHEN = "enabledWhen"; //$NON-NLS-1$ |
| |
| private String pageId; |
| |
| /** |
| * The list of subpages (immediate children) of this node (element type: |
| * <code>RegistryPageContributor</code>). |
| */ |
| private Collection subPages = new ArrayList(); |
| |
| private boolean adaptable = false; |
| |
| private IConfigurationElement pageElement; |
| |
| private SoftReference filterProperties; |
| |
| private Expression enablementExpression; |
| |
| /** |
| * PropertyPageContributor constructor. |
| * |
| * @param pageId |
| * the id |
| * @param element |
| * the element |
| */ |
| public RegistryPageContributor(String pageId, IConfigurationElement element) { |
| this.pageId = pageId; |
| this.pageElement = element; |
| adaptable = Boolean |
| .valueOf( |
| pageElement |
| .getAttribute(PropertyPagesRegistryReader.ATT_ADAPTABLE)) |
| .booleanValue(); |
| initializeEnablement(element); |
| } |
| |
| /* |
| * (non-Javadoc) |
| * |
| * @see org.eclipse.ui.internal.dialogs.IPropertyPageContributor#contributePropertyPages(org.eclipse.ui.internal.dialogs.PropertyPageManager, |
| * java.lang.Object) |
| */ |
| public boolean contributePropertyPages(PropertyPageManager mng, |
| Object element) { |
| PropertyPageNode node = new PropertyPageNode(this, element); |
| |
| if (getCategory() == null) { |
| mng.addToRoot(node); |
| return true; |
| } |
| if (!mng.addToDeep(getCategory(), node)) |
| mng.addToRoot(node); |
| |
| return true; |
| } |
| |
| /** |
| * Creates the page based on the information in the configuration element. |
| * |
| * @param element |
| * the adaptable element |
| * @return the property page |
| * @throws CoreException |
| * thrown if there is a problem creating the apge |
| */ |
| public IWorkbenchPropertyPage createPage(Object element) |
| throws CoreException { |
| IWorkbenchPropertyPage ppage = null; |
| ppage = (IWorkbenchPropertyPage) WorkbenchPlugin.createExtension( |
| pageElement, IWorkbenchRegistryConstants.ATT_CLASS); |
| |
| ppage.setTitle(getPageName()); |
| |
| Object adapted = element; |
| if (adaptable) { |
| adapted = getAdaptedElement(element); |
| if (adapted == null) { |
| String message = "Error adapting selection to Property page " + pageId + " is being ignored"; //$NON-NLS-1$ //$NON-NLS-2$ |
| throw new CoreException(new Status(IStatus.ERROR, |
| WorkbenchPlugin.PI_WORKBENCH, IStatus.ERROR, message, |
| null)); |
| } |
| } |
| |
| if (adapted instanceof IAdaptable) |
| ppage.setElement((IAdaptable) adapted); |
| else |
| ppage.setElement(new AdaptableForwarder(adapted)); |
| |
| return ppage; |
| } |
| |
| /** |
| * Find an adapted element from the receiver. |
| * |
| * @param element |
| * @return the adapted element or <code>null</code> if it could not be |
| * found. |
| */ |
| private Object getAdaptedElement(Object element) { |
| Object adapted = LegacyResourceSupport.getAdapter(element, |
| getObjectClass()); |
| if (adapted != null) |
| return adapted; |
| |
| return null; |
| } |
| |
| /** |
| * Return the object class name |
| * |
| * @return the object class name |
| */ |
| public String getObjectClass() { |
| return pageElement |
| .getAttribute(PropertyPagesRegistryReader.ATT_OBJECTCLASS); |
| } |
| |
| /** |
| * Returns page icon as defined in the registry. |
| * |
| * @return the page icon |
| */ |
| public ImageDescriptor getPageIcon() { |
| String iconName = pageElement |
| .getAttribute(IWorkbenchRegistryConstants.ATT_ICON); |
| if (iconName == null) |
| return null; |
| return AbstractUIPlugin.imageDescriptorFromPlugin(pageElement |
| .getNamespaceIdentifier(), iconName); |
| } |
| |
| /** |
| * Returns page ID as defined in the registry. |
| * |
| * @return the page id |
| */ |
| |
| public String getPageId() { |
| return pageId; |
| } |
| |
| /** |
| * Returns page name as defined in the registry. |
| * |
| * @return the page name |
| */ |
| public String getPageName() { |
| return pageElement.getAttribute(IWorkbenchRegistryConstants.ATT_NAME); |
| } |
| |
| /** |
| * Return true if name filter is not defined in the registry for this page, |
| * or if name of the selected object matches the name filter. |
| */ |
| public boolean isApplicableTo(Object object) { |
| |
| if (failsEnablement(object)) |
| return false; |
| |
| // Test name filter |
| String nameFilter = pageElement |
| .getAttribute(PropertyPagesRegistryReader.ATT_NAME_FILTER); |
| if (nameFilter != null) { |
| String objectName = object.toString(); |
| IWorkbenchAdapter adapter = (IWorkbenchAdapter) Util.getAdapter(object, |
| IWorkbenchAdapter.class); |
| if (adapter != null) { |
| String elementName = adapter.getLabel(object); |
| if (elementName != null) { |
| objectName = elementName; |
| } |
| } |
| if (!SelectionEnabler.verifyNameMatch(objectName, nameFilter)) |
| return false; |
| } |
| |
| // Test custom filter |
| if (getFilterProperties() == null) |
| return true; |
| IActionFilter filter = null; |
| |
| // Do the free IResource adapting |
| Object adaptedObject = LegacyResourceSupport.getAdaptedResource(object); |
| if (adaptedObject != null) { |
| object = adaptedObject; |
| } |
| |
| filter = (IActionFilter)Util.getAdapter(object, IActionFilter.class); |
| |
| if (filter != null) |
| return testCustom(object, filter); |
| |
| return true; |
| } |
| |
| /** |
| * Return whether or not object fails the enablement criterea. |
| * |
| * @param object |
| * @return boolean <code>true</code> if it fails the enablement test |
| */ |
| private boolean failsEnablement(Object object) { |
| if (enablementExpression == null) |
| return false; |
| try { |
| return enablementExpression.evaluate( |
| new EvaluationContext(null, object)).equals( |
| EvaluationResult.FALSE); |
| } catch (CoreException e) { |
| WorkbenchPlugin.log(e); |
| return false; |
| } |
| } |
| |
| /** |
| * Initialize the enablement expression for this decorator |
| */ |
| protected void initializeEnablement(IConfigurationElement definingElement) { |
| IConfigurationElement[] elements = definingElement |
| .getChildren(CHILD_ENABLED_WHEN); |
| |
| if (elements.length == 0) |
| return; |
| |
| try { |
| IConfigurationElement[] enablement = elements[0].getChildren(); |
| if (enablement.length == 0) |
| return; |
| enablementExpression = ExpressionConverter.getDefault().perform( |
| enablement[0]); |
| } catch (CoreException e) { |
| WorkbenchPlugin.log(e); |
| } |
| |
| } |
| |
| /** |
| * Returns whether the object passes a custom key value filter implemented |
| * by a matcher. |
| */ |
| private boolean testCustom(Object object, IActionFilter filter) { |
| Map filterProperties = getFilterProperties(); |
| |
| if (filterProperties == null) |
| return false; |
| Iterator iter = filterProperties.keySet().iterator(); |
| while (iter.hasNext()) { |
| String key = (String) iter.next(); |
| String value = (String) filterProperties.get(key); |
| if (!filter.testAttribute(object, key, value)) |
| return false; |
| } |
| return true; |
| } |
| |
| /* |
| * @see IObjectContributor#canAdapt() |
| */ |
| public boolean canAdapt() { |
| return adaptable; |
| } |
| |
| /** |
| * Get the id of the category. |
| * |
| * @return String |
| * @since 3.1 |
| */ |
| public String getCategory() { |
| return pageElement |
| .getAttribute(CategorizedPageRegistryReader.ATT_CATEGORY); |
| } |
| |
| /** |
| * Return the children of the receiver. |
| * |
| * @return Collection |
| */ |
| public Collection getSubPages() { |
| return subPages; |
| } |
| |
| /** |
| * Add child to the list of children. |
| * |
| * @param child |
| */ |
| public void addSubPage(RegistryPageContributor child) { |
| subPages.add(child); |
| } |
| |
| private Map getFilterProperties() { |
| if (filterProperties == null || filterProperties.get() == null) { |
| Map map = new HashMap(); |
| filterProperties = new SoftReference(map); |
| IConfigurationElement[] children = pageElement.getChildren(); |
| for (int i = 0; i < children.length; i++) { |
| processChildElement(map, children[i]); |
| } |
| } |
| return (Map) filterProperties.get(); |
| } |
| |
| /** |
| * Get the child with the given id. |
| * |
| * @param id |
| * @return RegistryPageContributor |
| */ |
| public Object getChild(String id) { |
| Iterator iterator = subPages.iterator(); |
| while (iterator.hasNext()) { |
| RegistryPageContributor next = (RegistryPageContributor) iterator |
| .next(); |
| if (next.getPageId().equals(id)) |
| return next; |
| } |
| return null; |
| } |
| |
| /** |
| * Parses child element and processes it. |
| * |
| * @since 3.1 |
| */ |
| private void processChildElement(Map map, IConfigurationElement element) { |
| String tag = element.getName(); |
| if (tag.equals(PropertyPagesRegistryReader.TAG_FILTER)) { |
| String key = element |
| .getAttribute(PropertyPagesRegistryReader.ATT_FILTER_NAME); |
| String value = element |
| .getAttribute(PropertyPagesRegistryReader.ATT_FILTER_VALUE); |
| if (key == null || value == null) |
| return; |
| map.put(key, value); |
| } |
| } |
| |
| /* |
| * (non-Javadoc) |
| * |
| * @see org.eclipse.core.runtime.IAdaptable#getAdapter(java.lang.Class) |
| * @since 3.1 |
| */ |
| public Object getAdapter(Class adapter) { |
| if (adapter.equals(IConfigurationElement.class)) { |
| return getConfigurationElement(); |
| } |
| return null; |
| } |
| |
| /** |
| * @return the configuration element |
| * @since 3.1 |
| */ |
| IConfigurationElement getConfigurationElement() { |
| return pageElement; |
| } |
| } |