| /******************************************************************************* |
| * Copyright (c) 2000, 2010 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 |
| * Nikolay Botev - bug 240651 |
| *******************************************************************************/ |
| |
| package org.eclipse.ui.part; |
| |
| import org.eclipse.core.runtime.IProgressMonitor; |
| import org.eclipse.swt.widgets.Composite; |
| import org.eclipse.ui.IEditorInput; |
| import org.eclipse.ui.IEditorPart; |
| import org.eclipse.ui.IEditorReference; |
| import org.eclipse.ui.IEditorSite; |
| import org.eclipse.ui.IPartListener2; |
| import org.eclipse.ui.IPropertyListener; |
| import org.eclipse.ui.IWorkbenchPart; |
| import org.eclipse.ui.IWorkbenchPartReference; |
| import org.eclipse.ui.PartInitException; |
| import org.eclipse.ui.internal.EditorSite; |
| import org.eclipse.ui.internal.PartService; |
| import org.eclipse.ui.internal.PartSite; |
| import org.eclipse.ui.internal.WorkbenchPage; |
| |
| /** |
| * A AbstractMultiEditor is a composite of editors. |
| * |
| * This class is intended to be subclassed. |
| * |
| * @since 1.4 |
| */ |
| public abstract class AbstractMultiEditor extends EditorPart { |
| |
| private int activeEditorIndex; |
| |
| private IEditorPart innerEditors[]; |
| |
| private IPartListener2 propagationListener; |
| |
| /** |
| * Constructs an editor to contain other editors. |
| */ |
| public AbstractMultiEditor() { |
| super(); |
| } |
| |
| /** |
| * Handles a property change notification from a nested editor. The default |
| * implementation simply forwards the change to listeners on this multi |
| * editor by calling <code>firePropertyChange</code> with the same property |
| * id. For example, if the dirty state of a nested editor changes (property |
| * id <code>ISaveablePart.PROP_DIRTY</code>), this method handles it by |
| * firing a property change event for <code>ISaveablePart.PROP_DIRTY</code> |
| * to property listeners on this multi editor. |
| * <p> |
| * Subclasses may extend or reimplement this method. |
| * </p> |
| * |
| * @param propId |
| * the id of the property that changed |
| * @since 1.4 |
| */ |
| protected void handlePropertyChange(int propId) { |
| firePropertyChange(propId); |
| } |
| |
| /* |
| * @see IEditorPart#doSave(IProgressMonitor) |
| */ |
| public void doSave(IProgressMonitor monitor) { |
| for (int i = 0; i < innerEditors.length; i++) { |
| IEditorPart e = innerEditors[i]; |
| e.doSave(monitor); |
| } |
| } |
| |
| /* |
| * @see IEditorPart#doSaveAs() |
| */ |
| public void doSaveAs() { |
| //no-op |
| } |
| |
| /* |
| * @see IEditorPart#init(IEditorSite, IEditorInput) |
| */ |
| public void init(IEditorSite site, IEditorInput input) |
| throws PartInitException { |
| init(site, (MultiEditorInput) input); |
| } |
| |
| /** |
| * @param site |
| * @param input |
| * @throws PartInitException |
| * |
| * @see IEditorPart#init(IEditorSite, IEditorInput) |
| */ |
| public void init(IEditorSite site, MultiEditorInput input) |
| throws PartInitException { |
| setInput(input); |
| setSite(site); |
| setPartName(input.getName()); |
| setTitleToolTip(input.getToolTipText()); |
| setupEvents(); |
| } |
| |
| /* |
| * @see IEditorPart#isDirty() |
| */ |
| public boolean isDirty() { |
| for (int i = 0; i < innerEditors.length; i++) { |
| IEditorPart e = innerEditors[i]; |
| if (e.isDirty()) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| /* |
| * @see IEditorPart#isSaveAsAllowed() |
| */ |
| public boolean isSaveAsAllowed() { |
| return false; |
| } |
| |
| /* |
| * @see IWorkbenchPart#setFocus() |
| */ |
| public void setFocus() { |
| innerEditors[activeEditorIndex].setFocus(); |
| } |
| |
| /** |
| * Returns the active inner editor. |
| * @return the active editor |
| */ |
| public final IEditorPart getActiveEditor() { |
| return innerEditors[activeEditorIndex]; |
| } |
| |
| /** |
| * Returns an array with all inner editors. |
| * @return the inner editors |
| */ |
| public final IEditorPart[] getInnerEditors() { |
| return innerEditors; |
| } |
| |
| /** |
| * Set the inner editors. |
| * |
| * Should not be called by clients. |
| * |
| * @param children |
| * the inner editors of this multi editor |
| * @noreference This method is not intended to be referenced by clients. |
| */ |
| public final void setChildren(IEditorPart[] children) { |
| innerEditors = children; |
| activeEditorIndex = 0; |
| |
| for (int i = 0; i < children.length; i++) { |
| children[i].addPropertyListener( new IPropertyListener() { |
| public void propertyChanged(Object source, int propId) { |
| handlePropertyChange(propId); |
| } |
| }); |
| } |
| |
| innerEditorsCreated(); |
| } |
| |
| /** |
| * Called as soon as the inner editors have been created and are available. |
| */ |
| protected abstract void innerEditorsCreated(); |
| |
| /** |
| * Activates the given nested editor. |
| * |
| * @param part the nested editor |
| * @since 1.4 |
| */ |
| public void activateEditor(IEditorPart part) { |
| activeEditorIndex = getIndex(part); |
| IEditorPart e = getActiveEditor(); |
| EditorSite innerSite = (EditorSite) e.getEditorSite(); |
| ((WorkbenchPage) innerSite.getPage()).requestActivation(e); |
| } |
| |
| /** |
| * Returns the index of the given nested editor. |
| * |
| * @return the index of the nested editor |
| * @since 1.4 |
| */ |
| protected int getIndex(IEditorPart editor) { |
| for (int i = 0; i < innerEditors.length; i++) { |
| if (innerEditors[i] == editor) { |
| return i; |
| } |
| } |
| return -1; |
| } |
| |
| /** |
| * Set up the AbstractMultiEditor to propagate events like partClosed(). |
| * |
| * @since 1.4 |
| */ |
| private void setupEvents() { |
| propagationListener = new IPartListener2() { |
| public void partActivated(IWorkbenchPartReference partRef) { |
| } |
| |
| public void partBroughtToTop(IWorkbenchPartReference partRef) { |
| } |
| |
| public void partClosed(IWorkbenchPartReference partRef) { |
| IWorkbenchPart part = partRef.getPart(false); |
| if (part == AbstractMultiEditor.this && innerEditors != null) { |
| PartService partService = ((WorkbenchPage) getSite() |
| .getPage()).getPartService(); |
| for (int i = 0; i < innerEditors.length; i++) { |
| IEditorPart editor = innerEditors[i]; |
| IWorkbenchPartReference innerRef = ((PartSite) editor |
| .getSite()).getPartReference(); |
| partService.firePartClosed(innerRef); |
| } |
| } |
| } |
| |
| public void partDeactivated(IWorkbenchPartReference partRef) { |
| } |
| |
| public void partOpened(IWorkbenchPartReference partRef) { |
| IWorkbenchPart part = partRef.getPart(false); |
| if (part == AbstractMultiEditor.this && innerEditors != null) { |
| PartService partService = ((WorkbenchPage) getSite() |
| .getPage()).getPartService(); |
| for (int i = 0; i < innerEditors.length; i++) { |
| IEditorPart editor = innerEditors[i]; |
| IWorkbenchPartReference innerRef = ((PartSite) editor |
| .getSite()).getPartReference(); |
| partService.firePartOpened(innerRef); |
| } |
| } |
| } |
| |
| public void partHidden(IWorkbenchPartReference partRef) { |
| } |
| |
| public void partVisible(IWorkbenchPartReference partRef) { |
| } |
| |
| public void partInputChanged(IWorkbenchPartReference partRef) { |
| } |
| }; |
| getSite().getPage().addPartListener(propagationListener); |
| } |
| |
| /** |
| * Release the added listener. |
| * |
| * @since 1.4 |
| */ |
| public void dispose() { |
| getSite().getPage().removePartListener(propagationListener); |
| super.dispose(); |
| } |
| |
| /** |
| * This method is called after createPartControl has been executed and |
| * should return the container for the given inner editor. |
| * |
| * @param innerEditorReference |
| * a reference to the inner editor that is being created. |
| * @return the container in which the inner editor's pane and part controls |
| * are to be created. |
| */ |
| public abstract Composite getInnerEditorContainer(IEditorReference innerEditorReference); |
| |
| } |