/****************************************************************************** * Copyright (c) 2002, 2009 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.gmf.runtime.diagram.ui.commands; import java.util.ArrayList; import java.util.HashSet; import java.util.Iterator; import java.util.List; import java.util.ListIterator; import java.util.Set; import org.eclipse.core.commands.ExecutionException; import org.eclipse.core.runtime.Assert; import org.eclipse.core.runtime.IAdaptable; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.draw2d.geometry.Dimension; import org.eclipse.draw2d.geometry.Point; import org.eclipse.draw2d.geometry.Rectangle; import org.eclipse.emf.ecore.EAnnotation; import org.eclipse.emf.transaction.TransactionalEditingDomain; import org.eclipse.gmf.runtime.common.core.command.CommandResult; import org.eclipse.gmf.runtime.common.ui.util.ICustomData; import org.eclipse.gmf.runtime.diagram.core.util.ViewUtil; import org.eclipse.gmf.runtime.diagram.ui.internal.commands.ClipboardCommand; import org.eclipse.gmf.runtime.diagram.ui.util.MeasurementUnitHelper; import org.eclipse.gmf.runtime.draw2d.ui.mapmode.IMapMode; import org.eclipse.gmf.runtime.emf.clipboard.core.ClipboardUtil; import org.eclipse.gmf.runtime.notation.Bendpoints; import org.eclipse.gmf.runtime.notation.Edge; import org.eclipse.gmf.runtime.notation.LayoutConstraint; import org.eclipse.gmf.runtime.notation.Location; import org.eclipse.gmf.runtime.notation.MeasurementUnit; import org.eclipse.gmf.runtime.notation.Node; import org.eclipse.gmf.runtime.notation.NotationPackage; import org.eclipse.gmf.runtime.notation.RelativeBendpoints; import org.eclipse.gmf.runtime.notation.Size; import org.eclipse.gmf.runtime.notation.View; import org.eclipse.gmf.runtime.notation.datatype.RelativeBendpoint; /** * Paste Command for the views * * @author Vishy Ramaswamy */ public class PasteCommand extends ClipboardCommand { /** * The clipboard data */ private final ICustomData[] data; private IMapMode mm; /** * Constructor for PasteCommand. * @param editingDomain * the editing domain through which model changes are made * @param label * @param viewContext * @param data * @param mm * the <code>IMapMode</code> that is used to convert the layout constraint * and calculate the offset in logical coordinates */ public PasteCommand(TransactionalEditingDomain editingDomain, String label, View viewContext, ICustomData[] data, IMapMode mm) { super(editingDomain, label, viewContext); Assert.isNotNull(data); this.data = data; this.mm = mm; } protected CommandResult doExecuteWithResult(IProgressMonitor monitor, IAdaptable info) throws ExecutionException { /* Paste on the target */ if (data != null && data.length > 0) { List allViews = new ArrayList(); for (int j = 0; j < data.length; j++) { /* Get the string from the clipboard data */ String xml = new String(data[j].getData()); /* Paste the xml on to the target view's diagram */ List views = pasteFromString(getViewContext(),xml); allViews.addAll(views); } return CommandResult.newOKCommandResult(allViews); } return CommandResult.newOKCommandResult(); } /** * Method pasteFromString. * pastes the clipboard contents on to self * @param clipboard The clipboard contents - serialization used during copy * @return List The list of IView resulting from the paste */ protected List pasteFromString(View view, String clipboard) { ArrayList retval = new ArrayList(); Iterator pastedElements = ClipboardUtil.pasteElementsFromString(clipboard, view, null, null).iterator(); // get the measurement unit MeasurementUnit mu = MeasurementUnit.HIMETRIC_LITERAL; while( pastedElements.hasNext() ) { Object element = pastedElements.next(); if (element instanceof View) { retval.add(element); } else if (element instanceof EAnnotation) { EAnnotation measureUnitAnnotation = (EAnnotation)element; String unitName = measureUnitAnnotation.getSource(); mu = MeasurementUnit.get(unitName); view.getEAnnotations().remove(element); } } /* Set the new bounds for the pasted IShapeView views */ Set edges = convertNodesConstraint(retval, mu, true); // now go through all associated edges and adjust the bendpoints convertEdgeBendpoints(mu, edges); return retval; } /** * @param mu the <code>MeasurementUnit</code> for the notation diagram. * @param edges the <code>Set</code> of edges to convert the bendpoints of. */ private void convertEdgeBendpoints(MeasurementUnit mu, Set edges) { for (Edge nextEdge : (Set<Edge>) edges) { Bendpoints bendpoints = nextEdge.getBendpoints(); if (bendpoints instanceof RelativeBendpoints) { RelativeBendpoints relBendpoints = (RelativeBendpoints)bendpoints; List points = relBendpoints.getPoints(); List newpoints = new ArrayList(points.size()); ListIterator li = points.listIterator(); IMapMode viewMapMode = MeasurementUnitHelper.getMapMode(mu); while (li.hasNext()) { RelativeBendpoint rb = (RelativeBendpoint)li.next(); Dimension source = new Dimension(rb.getSourceX(), rb.getSourceY()); Dimension target = new Dimension(rb.getTargetX(), rb.getTargetY()); if (!viewMapMode.equals(mm)) { source = (Dimension)viewMapMode.LPtoDP(source); source = (Dimension)mm.DPtoLP(source); target = (Dimension)viewMapMode.LPtoDP(target); target = (Dimension)mm.DPtoLP(target); } newpoints.add(new RelativeBendpoint(source.width, source.height, target.width, target.height)); } relBendpoints.setPoints(newpoints); } } } /** * @param retval the <code>List</code> of <code>Node</code> objects to convert the constraint of. * @param mu the <code>MeasurementUnit</code> for the notation diagram. * @return the <code>Set</code> of <code>Edge</code> views that are attached to the list of nodes */ private Set convertNodesConstraint(List retval, MeasurementUnit mu, boolean isProcessOffset) { Set edges = new HashSet(); for (Iterator i = retval.iterator(); i.hasNext();) { View nextView = (View) i.next(); if (nextView instanceof Node) { Node node = (Node)nextView; Point loc = new Point(0, 0); LayoutConstraint lc = node.getLayoutConstraint(); if (lc instanceof Location) { Location locC = (Location)lc; loc = new Point(locC.getX(), locC.getY()); } Dimension size = new Dimension(0, 0); if (lc instanceof Size) { Size sizeC = (Size)lc; size = new Dimension(sizeC.getWidth(), sizeC.getHeight()); } IMapMode viewMapMode = MeasurementUnitHelper.getMapMode(mu); if (!viewMapMode.equals(mm)) { // convert location to native coordinates loc = (Point)viewMapMode.LPtoDP(loc); loc = (Point)mm.DPtoLP(loc); // convert size to native coordinates Dimension origSize = new Dimension(size); size = (Dimension)viewMapMode.LPtoDP(size); size = (Dimension)mm.DPtoLP(size); if (origSize.width == -1) size.width = -1; if (origSize.height == -1) size.height = -1; } Rectangle constraintRect = new Rectangle(loc, size); if ( isProcessOffset ) { constraintRect = processNodeOffset(node, constraintRect); } ViewUtil.setStructuralFeatureValue(nextView,NotationPackage.eINSTANCE.getLocation_X(), Integer.valueOf(constraintRect.x)); ViewUtil.setStructuralFeatureValue(nextView,NotationPackage.eINSTANCE.getLocation_Y(), Integer.valueOf(constraintRect.y)); ViewUtil.setStructuralFeatureValue(nextView,NotationPackage.eINSTANCE.getSize_Width(), Integer.valueOf(constraintRect.width)); ViewUtil.setStructuralFeatureValue(nextView,NotationPackage.eINSTANCE.getSize_Height(), Integer.valueOf(constraintRect.height)); edges.addAll(ViewUtil.getTargetConnections(nextView)); edges.addAll(ViewUtil.getSourceConnections(nextView)); // recursively perform the same operation on children of the node if (node.eIsSet(NotationPackage.eINSTANCE.getView_PersistedChildren())) { edges.addAll(convertNodesConstraint(node.getPersistedChildren(), mu, false)); } } } return edges; } /** * Add offset node position. * @param node * @param constraintRect * @return */ protected Rectangle processNodeOffset(Node node, Rectangle constraintRect) { return constraintRect.getTranslated(mm.DPtoLP(10), mm.DPtoLP(10)); } /** * Get map mode associated with this Paste command. * @return */ public IMapMode getMapMode() { return mm; } }