From 403cf22e3a9df117ba32314afd15515556925234 Mon Sep 17 00:00:00 2001 From: Laurent Redor Date: Tue, 16 Apr 2024 18:27:20 +0200 Subject: [PATCH] [359] New implementation of move with arrow keys Bug: https://github.com/eclipse-sirius/sirius-desktop/issues/359 --- .../palette/SiriusSelectionToolEx.java | 100 +++++- .../internal/ruler/SiriusSnapToGeometry.java | 80 ++++- .../internal/ruler/SiriusSnapToGridEx.java | 100 ++++++ .../ruler/SiriusSnapToHelperUtil.java | 5 +- .../ui/SnapToAllDragEditPartsTracker.java | 325 +++++++++++++++++- 5 files changed, 596 insertions(+), 14 deletions(-) create mode 100644 plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToGridEx.java diff --git a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/palette/SiriusSelectionToolEx.java b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/palette/SiriusSelectionToolEx.java index 60c1e93e3b..0b6116d0fb 100644 --- a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/palette/SiriusSelectionToolEx.java +++ b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/palette/SiriusSelectionToolEx.java @@ -1,5 +1,5 @@ /******************************************************************************* - * Copyright (c) 2017 THALES GLOBAL SERVICES. + * Copyright (c) 2017, 2024 THALES GLOBAL SERVICES. * 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 @@ -12,14 +12,23 @@ *******************************************************************************/ package org.eclipse.sirius.diagram.ui.tools.internal.palette; +import java.util.Optional; + +import org.eclipse.draw2d.XYLayout; +import org.eclipse.gef.DragTracker; import org.eclipse.gef.EditPart; +import org.eclipse.gef.EditPolicy; +import org.eclipse.gef.GraphicalEditPart; +import org.eclipse.gef.editpolicies.NonResizableEditPolicy; import org.eclipse.gmf.runtime.diagram.ui.internal.editparts.NoteAttachmentEditPart; +import org.eclipse.gmf.runtime.diagram.ui.internal.figures.BorderItemContainerFigure; import org.eclipse.gmf.runtime.diagram.ui.services.palette.SelectionToolEx; import org.eclipse.sirius.diagram.ui.edit.api.part.AbstractDiagramEdgeEditPart; import org.eclipse.sirius.diagram.ui.internal.edit.parts.AbstractDEdgeNameEditPart; import org.eclipse.sirius.diagram.ui.internal.edit.parts.DNodeListElementEditPart; import org.eclipse.sirius.diagram.ui.internal.edit.parts.DNodeNameEditPart; import org.eclipse.sirius.diagram.ui.tools.internal.part.SiriusDiagramGraphicalViewer; +import org.eclipse.swt.events.KeyEvent; /** * Specific Sirius SelectionToolEx to use findMouseEventTargetAt instead of findObjectAtExcluding. This allows to @@ -76,4 +85,93 @@ private boolean isSiriusSpecificEditPart(EditPart editPart) { return editPart instanceof DNodeNameEditPart || editPart instanceof AbstractDiagramEdgeEditPart || editPart instanceof AbstractDEdgeNameEditPart || editPart instanceof DNodeListElementEditPart; } + + + /** + * Method overridden to allow arrow key press with modifier. + * + * @see org.eclipse.gmf.runtime.diagram.ui.services.palette.SelectionToolEx#handleKeyDown(org.eclipse.swt.events.KeyEvent) + */ + @Override + protected boolean handleKeyDown(KeyEvent e) { + Optional optionalLocalresult = specificHandleKeyDown(e); + if (optionalLocalresult.isEmpty()) { + return super.handleKeyDown(e); + } else { + return optionalLocalresult.get().booleanValue(); + } + } + + /** + * Method inspired by org.eclipse.gmf.runtime.diagram.ui.services.palette.SelectionToolEx.handleKeyDown(KeyEvent) to + * authorize arrow keys with modifiers ({@link #acceptArrowKey(KeyEvent)} instead of + * {@link SelectionToolEx#acceptArrowKeyOnly(KeyEvent)}; Mainly for the "Alt" modifier to disable the snap. + * + * @see org.eclipse.gmf.runtime.diagram.ui.services.palette.SelectionToolEx#handleKeyDown(org.eclipse.swt.events.KeyEvent) + */ + protected Optional specificHandleKeyDown(KeyEvent e) { + Optional optionalLocalresult = Optional.empty(); + if (acceptArrowKey(e) && getState() == STATE_INITIAL && !getCurrentViewer().getSelectedEditParts().isEmpty()) { + + EditPart selectedEP = (EditPart) getCurrentViewer().getSelectedEditParts().get(0); + + if (selectedEP instanceof GraphicalEditPart) { + + GraphicalEditPart gep = (GraphicalEditPart) selectedEP; + + /* + * The shape we'll be moved in the direction of the arrow key if: 1) It has the appropriate edit policy + * that supports shape moving installed on the editpart 2) The editparts figure's parent layout manager + * is some sort of XYLayout In all other cases we just change the selection based on arrow key + * (implemented in GEF). + */ + if (gep.getEditPolicy(EditPolicy.PRIMARY_DRAG_ROLE) instanceof NonResizableEditPolicy && gep.getFigure().getParent() != null + && (gep.getFigure().getParent().getLayoutManager() instanceof XYLayout || gep.getFigure().getParent() instanceof BorderItemContainerFigure)) { + + resetHover(); + + if (getDragTracker() != null) { + getDragTracker().deactivate(); + } + + setState(STATE_ACCESSIBLE_DRAG_IN_PROGRESS); + + setTargetEditPart(gep); + + updateTargetRequest(); + DragTracker dragTracker = gep.getDragTracker(getTargetRequest()); + if (dragTracker != null) { + setDragTracker(dragTracker); + dragTracker.keyDown(e, getCurrentViewer()); + lockTargetEditPart(gep); + optionalLocalresult = Optional.of(true); + } + optionalLocalresult = Optional.of(false); + } + } + } + return optionalLocalresult; + } + + /** + * As for method {@link #handleKeyDown(KeyEvent)}, this method is overridden to "finish" the move in case of arrow + * key press. + * + * @see org.eclipse.gmf.runtime.diagram.ui.services.palette.SelectionToolEx#handleKeyUp(org.eclipse.swt.events.KeyEvent) + */ + @Override + protected boolean handleKeyUp(KeyEvent e) { + boolean returnVal = super.handleKeyUp(e); + if (acceptArrowKey(e)) { + // In superclass SelectionToolEx.handleKeyUp(KeyEvent), it was "if (acceptArrowKeyOnly(e) && + // !isUsingTraverseHandles) {". + if (getDragTracker() != null) { + getDragTracker().commitDrag(); + } + setDragTracker(null); + setState(STATE_INITIAL); + unlockTargetEditPart(); + } + return returnVal; + } } diff --git a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToGeometry.java b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToGeometry.java index bc1915ca2e..e9cb6ff348 100644 --- a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToGeometry.java +++ b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToGeometry.java @@ -1,5 +1,5 @@ /******************************************************************************* - * Copyright (c) 2015, 2016 THALES GLOBAL SERVICES. + * Copyright (c) 2015, 2024 THALES GLOBAL SERVICES. * 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 @@ -15,12 +15,17 @@ import java.util.ArrayList; import java.util.Iterator; import java.util.List; +import java.util.Optional; +import org.eclipse.draw2d.IFigure; +import org.eclipse.draw2d.geometry.Point; import org.eclipse.draw2d.geometry.PrecisionRectangle; import org.eclipse.draw2d.geometry.Rectangle; +import org.eclipse.draw2d.geometry.Translatable; import org.eclipse.gef.EditPart; import org.eclipse.gef.GraphicalEditPart; import org.eclipse.gef.Request; +import org.eclipse.gef.handles.HandleBounds; import org.eclipse.gmf.runtime.diagram.ui.editparts.IGraphicalEditPart; import org.eclipse.gmf.runtime.diagram.ui.internal.ruler.SnapToGeometryEx; import org.eclipse.sirius.diagram.ui.edit.api.part.AbstractBorderedDiagramElementEditPart; @@ -29,6 +34,8 @@ import org.eclipse.sirius.diagram.ui.graphical.edit.policies.SnapChangeBoundsRequest; import org.eclipse.sirius.diagram.ui.internal.edit.policies.SnapBendpointRequest; import org.eclipse.sirius.diagram.ui.tools.internal.ui.NoCopyDragEditPartsTrackerEx; +import org.eclipse.sirius.diagram.ui.tools.internal.ui.SnapToAllDragEditPartsTracker; +import org.eclipse.sirius.ext.draw2d.figure.FigureUtilities; import org.eclipse.sirius.ext.gef.query.EditPartQuery; import org.eclipse.sirius.ext.gmf.runtime.editparts.GraphicalHelper; @@ -46,6 +53,8 @@ public class SiriusSnapToGeometry extends SnapToGeometryEx { boolean snapToAll; + boolean isMoveWithArrowSiriusMode; + /** * A vertical or horizontal snapping point.
* Only overridden to have access to constructor. @@ -88,9 +97,22 @@ public int snapRectangle(Request request, int snapOrientation, PrecisionRectangl cols = null; } + // Get the new move mode with arrow keys from request (then used in makeRelative and makeAbsolute). Indeed, in + // this case, the absolute coordinate of baseRect consider the scrollbars. It is not the case + // otherwise. + setMoveWithArrowSiriusMode(((Boolean) (Optional.ofNullable(request.getExtendedData().get(SnapToAllDragEditPartsTracker.MOVE_WITH_ARROW_SIRIUS_MODE)).orElse(Boolean.FALSE))).booleanValue()); + return super.snapRectangle(request, snapOrientation, baseRect, result); } + protected void setMoveWithArrowSiriusMode(boolean moveWithArrowSiriusMode) { + this.isMoveWithArrowSiriusMode = moveWithArrowSiriusMode; + } + + protected boolean isMoveWithArrowSiriusMode() { + return isMoveWithArrowSiriusMode; + } + @Override protected List generateSnapPartsList(List exclusions) { if (!snapToAll) { @@ -182,4 +204,60 @@ protected void populateRowsAndCols(List parts) { } } } + + /** + * Translates from absolute to coordinates relative to the given figure. + * + * @param figure + * the reference figure + * @param t + * the object to translate + */ + @Override + protected void makeRelative(IFigure figure, Translatable t) { + // In the case of move with arrow key, there is nothing to do. Compute is already done in absolute coordinates + // (considering zoom and scrollbars). + if (!isMoveWithArrowSiriusMode()) { + super.makeRelative(figure, t); + } + } + + /** + * Translates from a given figure to absolute coordinates. + * + * @param figure + * the reference figure + * @param t + * the object to translate + */ + @Override + protected void makeAbsolute(IFigure figure, Translatable t) { + // In the case of move with arrow key, there is nothing to do. Compute is already done in absolute coordinates + // (considering zoom and scrollbars). + if (!isMoveWithArrowSiriusMode()) { + super.makeAbsolute(figure, t); + } + } + + // TODO: Comment and see if specific code is needed for border nodes in populateRowsAndCols(List) + @Override + protected Rectangle getFigureBounds(GraphicalEditPart part) { + if (isMoveWithArrowSiriusMode()) { + IFigure figure = part.getFigure(); + PrecisionRectangle bounds = null; + if (figure instanceof HandleBounds) { + bounds = new PrecisionRectangle(((HandleBounds) figure).getHandleBounds()); + } else { + bounds = new PrecisionRectangle(figure.getBounds()); + } + Point topLeft = bounds.getTopLeft(); + FigureUtilities.translateToAbsoluteByIgnoringScrollbar(figure, topLeft); + bounds.setX(topLeft.x); + bounds.setY(topLeft.y); + return bounds; + } else { + return super.getFigureBounds(part); + } + + } } diff --git a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToGridEx.java b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToGridEx.java new file mode 100644 index 0000000000..f98cb124be --- /dev/null +++ b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToGridEx.java @@ -0,0 +1,100 @@ +/******************************************************************************* + * Copyright (c) 2024 THALES GLOBAL SERVICES. + * 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: + * Obeo - initial API and implementation + *******************************************************************************/ +package org.eclipse.sirius.diagram.ui.tools.internal.ruler; + +import java.util.Optional; + +import org.eclipse.draw2d.IFigure; +import org.eclipse.draw2d.geometry.PrecisionRectangle; +import org.eclipse.draw2d.geometry.Translatable; +import org.eclipse.gef.GraphicalEditPart; +import org.eclipse.gef.Request; +import org.eclipse.gef.SnapToHelper; +import org.eclipse.gmf.runtime.diagram.ui.internal.ruler.SnapToGridEx; +import org.eclipse.sirius.diagram.ui.tools.internal.ui.SnapToAllDragEditPartsTracker; + +/** + * Overridden to support the new mode moving node with arrow keys. + * + * @author Laurent Redor + */ +@SuppressWarnings("restriction") +public class SiriusSnapToGridEx extends SnapToGridEx { + + boolean isMoveWithArrowSiriusMode; + + /** + * Default constructor. + * + * @param container + * the editpart which the grid is on + */ + public SiriusSnapToGridEx(GraphicalEditPart container) { + super(container); + } + + protected void setMoveWithArrowSiriusMode(boolean moveWithArrowSiriusMode) { + this.isMoveWithArrowSiriusMode = moveWithArrowSiriusMode; + } + + protected boolean isMoveWithArrowSiriusMode() { + return isMoveWithArrowSiriusMode; + } + + /** + * Method overridden to handle the new move mode with arrow keys (then used in makeRelative and makeAbsolute). + * Indeed, in this case, the absolute coordinate of rect consider the scrollbars. It is not the case + * otherwise. + * + * @see SnapToHelper#snapRectangle(Request, int, PrecisionRectangle, PrecisionRectangle) + */ + @Override + public int snapRectangle(Request request, int snapLocations, PrecisionRectangle rect, PrecisionRectangle result) { + setMoveWithArrowSiriusMode(((Boolean) (Optional.ofNullable(request.getExtendedData().get(SnapToAllDragEditPartsTracker.MOVE_WITH_ARROW_SIRIUS_MODE)).orElse(Boolean.FALSE))).booleanValue()); + return super.snapRectangle(request, snapLocations, rect, result); + } + + /** + * Translates from absolute to coordinates relative to the given figure. + * + * @param figure + * the reference figure + * @param t + * the object to translate + */ + @Override + protected void makeRelative(IFigure figure, Translatable t) { + // In the case of move with arrow key, there is nothing to do. Compute is already done in absolute coordinates + // (considering zoom and scrollbars). + if (!isMoveWithArrowSiriusMode()) { + super.makeRelative(figure, t); + } + } + + /** + * Translates from a given figure to absolute coordinates. + * + * @param figure + * the reference figure + * @param t + * the object to translate + */ + @Override + protected void makeAbsolute(IFigure figure, Translatable t) { + // In the case of move with arrow key, there is nothing to do. Compute is already done in absolute coordinates + // (considering zoom and scrollbars). + if (!isMoveWithArrowSiriusMode()) { + super.makeAbsolute(figure, t); + } + } +} diff --git a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToHelperUtil.java b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToHelperUtil.java index 9df17b611f..8eb3855fe8 100644 --- a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToHelperUtil.java +++ b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ruler/SiriusSnapToHelperUtil.java @@ -1,5 +1,5 @@ /****************************************************************************** - * Copyright (c) 2007, 2016 IBM Corporation and others. + * Copyright (c) 2007, 2024 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 @@ -24,7 +24,6 @@ import org.eclipse.gmf.runtime.diagram.ui.editparts.DiagramEditPart; import org.eclipse.gmf.runtime.diagram.ui.internal.editparts.ISurfaceEditPart; import org.eclipse.gmf.runtime.diagram.ui.internal.ruler.CompoundSnapToHelperEx; -import org.eclipse.gmf.runtime.diagram.ui.internal.ruler.SnapToGridEx; import org.eclipse.gmf.runtime.diagram.ui.internal.ruler.SnapToGuidesEx; import org.eclipse.gmf.runtime.diagram.ui.internal.ruler.SnapToHelperUtil; import org.eclipse.sirius.diagram.ui.edit.api.part.AbstractDiagramNodeEditPart; @@ -90,7 +89,7 @@ static public Object getSnapHelper(GraphicalEditPart editPart) { val = (Boolean) viewer.getProperty(SnapToGrid.PROPERTY_GRID_ENABLED); if (val != null && val.booleanValue()) { - snapStrategies.add(new SnapToGridEx(diagramEditPart)); + snapStrategies.add(new SiriusSnapToGridEx(diagramEditPart)); } if (snapStrategies.size() == 0) { diff --git a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ui/SnapToAllDragEditPartsTracker.java b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ui/SnapToAllDragEditPartsTracker.java index fb0325a29c..3a0531f209 100644 --- a/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ui/SnapToAllDragEditPartsTracker.java +++ b/plugins/org.eclipse.sirius.diagram.ui/src-diag/org/eclipse/sirius/diagram/ui/tools/internal/ui/SnapToAllDragEditPartsTracker.java @@ -1,5 +1,5 @@ /******************************************************************************* - * Copyright (c) 2015, 2016 THALES GLOBAL SERVICES. + * Copyright (c) 2015, 2024 THALES GLOBAL SERVICES. * 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 @@ -12,15 +12,29 @@ *******************************************************************************/ package org.eclipse.sirius.diagram.ui.tools.internal.ui; +import java.util.Date; +import java.util.List; + +import org.eclipse.core.runtime.Platform; import org.eclipse.draw2d.FigureCanvas; +import org.eclipse.draw2d.IFigure; +import org.eclipse.draw2d.PositionConstants; import org.eclipse.draw2d.geometry.Dimension; import org.eclipse.draw2d.geometry.Point; +import org.eclipse.draw2d.geometry.PrecisionPoint; +import org.eclipse.draw2d.geometry.PrecisionRectangle; import org.eclipse.gef.EditPart; import org.eclipse.gef.EditPartViewer; +import org.eclipse.gef.GraphicalEditPart; import org.eclipse.gef.SharedCursors; +import org.eclipse.gef.handles.HandleBounds; import org.eclipse.gef.requests.ChangeBoundsRequest; +import org.eclipse.gef.tools.AbstractTool; +import org.eclipse.gmf.runtime.diagram.ui.internal.ruler.SnapToHelperUtil; import org.eclipse.gmf.runtime.diagram.ui.tools.DragEditPartsTrackerEx; +import org.eclipse.sirius.ext.draw2d.figure.FigureUtilities; import org.eclipse.sirius.ext.gmf.runtime.diagram.ui.tools.MoveInDiagramDragTracker; +import org.eclipse.sirius.ext.gmf.runtime.editparts.GraphicalHelper; import org.eclipse.swt.SWT; import org.eclipse.swt.events.KeyEvent; import org.eclipse.swt.events.MouseEvent; @@ -35,8 +49,7 @@ public class SnapToAllDragEditPartsTracker extends DragEditPartsTrackerEx implements MoveInDiagramDragTracker { /** - * Constant passed to extended data of the request to keep the chosen mode - * (with KEY {@link #SNAP_TO_ALL}. + * Constant passed to extended data of the request to keep the chosen mode (with KEY {@link #SNAP_TO_ALL}). */ public static final String SNAP_TO_ALL_SHAPE_KEY = "snapToAllShape"; //$NON-NLS-1$ @@ -50,6 +63,45 @@ public class SnapToAllDragEditPartsTracker extends DragEditPartsTrackerEx implem */ public static final int SNAP_TO_ALL = SWT.F4; + /** + * Constant passed to extended data of the request to keep the chosen mode (with arrow key pressed). + */ + public static final String MOVE_WITH_ARROW_SIRIUS_MODE = "MoveWithArrowSiriusMode"; //$NON-NLS-1$ + + /** + * Copied from {@link AbstractTool#MODIFIER_NO_SNAPPING}. It is not accessible here but is used in the overridden of + * {@link #handleKeyDown(KeyEvent)}.
+ * Key modifier for ignoring snap while dragging. It's CTRL on Mac, and ALT on all other platforms. + */ + protected static final int MODIFIER_NO_SNAPPING; + + static { + if (Platform.OS_MACOSX.equals(Platform.getOS())) { + MODIFIER_NO_SNAPPING = SWT.CTRL; + } else { + MODIFIER_NO_SNAPPING = SWT.ALT; + } + } + + /** + * True when a move is triggered through an arrow key. This mode is a specific mode used by Sirius instead of the + * one of GMF/GEF. This "old" mode, based on a mouse move simulation, is "incomplete" when the container of the + * moved element has a scrollbar.
+ * This mode is enabled when an arrow key is pressed, in {@link #handleKeyDown(KeyEvent)} and is disabled when this + * key is released , in {@link #handleKeyUp(KeyEvent). + */ + private boolean moveWithArrowKeysSiriusMode; + + /** + * Copied from {@link org.eclipse.gef.tools.AbstractTool} to allow a similar acceleration move behavior. + */ + private long accessibleBegin; + + /** + * Copied from {@link org.eclipse.gef.tools.AbstractTool} to allow a similar acceleration move behavior. + */ + private int accessibleStep; + /** * The mode of this tracker concerning the snap to shape: *
    @@ -74,10 +126,10 @@ public SnapToAllDragEditPartsTracker(EditPart sourceEditPart) { } /** - * Overridden to update the {@link ChangeBoundsRequest} with information - * about snapToAll mode. + * Overridden to update the {@link ChangeBoundsRequest} with information about snapToAll mode and to adapt all code + * usually called in all "super.snapPoint()" for the new mode. * - * {@inheritDoc} + * @see org.eclipse.sirius.diagram.ui.tools.internal.ui.SnapToAllDragEditPartsTracker#snapPoint(org.eclipse.gef.requests.ChangeBoundsRequest) */ @Override protected void snapPoint(ChangeBoundsRequest request) { @@ -86,24 +138,148 @@ protected void snapPoint(ChangeBoundsRequest request) { } else { getTargetRequest().getExtendedData().put(SnapToAllDragEditPartsTracker.SNAP_TO_ALL_SHAPE_KEY, Boolean.FALSE); } - super.snapPoint(request); + if (!moveWithArrowKeysSiriusMode) { + super.snapPoint(request); + } else { + getTargetRequest().getExtendedData().put(SnapToAllDragEditPartsTracker.MOVE_WITH_ARROW_SIRIUS_MODE, Boolean.TRUE); + // Copied from + // org.eclipse.gmf.runtime.diagram.ui.tools.DragEditPartsTrackerEx.snapPoint(ChangeBoundsRequest) + // Adapted to force the snap in the expected initial direction but also to allow the snap in other + // perpendicular directions. + Point moveDelta = request.getMoveDelta(); + if (getState() == STATE_ACCESSIBLE_DRAG_IN_PROGRESS) { + int restrictedDirection = 0; + + if (moveDelta.preciseX() > 0) { + restrictedDirection = restrictedDirection | PositionConstants.EAST; + } else if (moveDelta.preciseX() < 0) { + restrictedDirection = restrictedDirection | PositionConstants.WEST; + } else { + restrictedDirection = restrictedDirection | PositionConstants.EAST_WEST; + } + + if (moveDelta.preciseY() > 0) { + restrictedDirection = restrictedDirection | PositionConstants.SOUTH; + } else if (moveDelta.preciseY() < 0) { + restrictedDirection = restrictedDirection | PositionConstants.NORTH; + } else { + restrictedDirection = restrictedDirection | PositionConstants.NORTH_SOUTH; + } + + request.getExtendedData().put(SnapToHelperUtil.RESTRICTED_DIRECTIONS, restrictedDirection); + } + double zoomFactor = GraphicalHelper.getZoom(getSourceEditPart()); + // Copied from org.eclipse.gef.tools.DragEditPartsTracker.snapPoint(ChangeBoundsRequest) to use "absolute + // without scroll bounds" instead of absolute coordinates + if (getSnapToHelper() != null && request.isSnapToEnabled()) { + PrecisionRectangle[] sourceAndCompoundSrcRectangle = captureSourceDimensionsWithAbsoluteBoundsAndWithoutScroll(); + PrecisionRectangle baseRect = sourceAndCompoundSrcRectangle[0].getPreciseCopy(); + PrecisionRectangle jointRect = sourceAndCompoundSrcRectangle[1].getPreciseCopy(); + // The zoom has been adapted in the request in "STATE_INITIAL" (statement "else" just + // bellow), we need to revert this adaptation for the base rect snap computation + PrecisionPoint preciseDelta = new PrecisionPoint(moveDelta.getScaled(1.d / zoomFactor)); + baseRect.translate(preciseDelta); + jointRect.translate(preciseDelta); + getSnapToHelper().snapPoint(request, PositionConstants.HORIZONTAL | PositionConstants.VERTICAL, new PrecisionRectangle[] { baseRect, jointRect }, preciseDelta); + request.setMoveDelta(preciseDelta.getScaled(GraphicalHelper.getZoom(getSourceEditPart()))); + } else if (getState() == STATE_INITIAL) { + // We are in a case of a move of 1 pixel. We need to adapt this pixel according to the zoom factor + // because later in + // org.eclipse.gef.editpolicies.ConstrainedLayoutEditPolicy.getConstraintFor(ChangeBoundsRequest, + // GraphicalEditPart) the compute is done with coordinates considering the zoom. + // This avoids to have, for example, a move of 4 pixels when the zoom level is 25% + request.setMoveDelta(new PrecisionPoint(moveDelta).getScaled(zoomFactor)); + } + } } + /** + * Method overridden to handle stapToAll and to enable the new mode in case of move with arrow key. + * + * @see org.eclipse.sirius.diagram.ui.tools.internal.ui.SnapToAllDragEditPartsTracker#handleKeyDown(org.eclipse.swt.events.KeyEvent) + */ @Override protected boolean handleKeyDown(KeyEvent event) { + boolean result = true; if (SnapToAllDragEditPartsTracker.SNAP_TO_ALL == event.keyCode) { snapToAllShape = !SnapToAllDragEditPartsTracker.DEFAULT_SNAP_TO_SHAPE_MODE; - return true; + } else if (!acceptArrowKey(event)) { + result = super.handleKeyDown(event); + } else { + moveWithArrowKeysSiriusMode = true; + // Enable the move behavior based on a request set according to the arrow direction pressed. + + // With this mode, it is not possible to enable the SnapToAllDragEditPartsTracker.SNAP_TO_ALL mode. Indeed, + // this mode is enabled by pressing the F4 key during a mouse move. It was also not available in the classic + // GMF/GEF mode. + + // By the way, the previous mode does not correctly handle the snap to shape (but it is currently not the + // goal of this new mode to fix this). + + // Reproduce the "acceleration behavior" of + // org.eclipse.gef.tools.DragEditPartsTracker.handleKeyDown(KeyEvent) to get the increment to use. + siriusAccStepIncrement(); + Point moveDelta; + switch (event.keyCode) { + case SWT.ARROW_DOWN: + moveDelta = new Point(0, siriusAccGetStep()); + break; + case SWT.ARROW_UP: + moveDelta = new Point(0, -siriusAccGetStep()); + break; + case SWT.ARROW_RIGHT: + int stepping = siriusAccGetStep(); + if (isCurrentViewerMirrored2()) { + stepping = -stepping; + } + moveDelta = new Point(stepping, 0); + break; + case SWT.ARROW_LEFT: + int step = -siriusAccGetStep(); + if (isCurrentViewerMirrored2()) { + step = -step; + } + moveDelta = new Point(step, 0); + break; + default: + moveDelta = new Point(0, 0); + } + // Set the request according to the move + ChangeBoundsRequest req = (ChangeBoundsRequest) getTargetRequest(); + req.setMoveDelta(req.getMoveDelta().getTranslated(moveDelta)); + req.setConstrainedResize(false); + // The org.eclipse.gef.tools.DragEditPartsTracker.MODIFIER_CONSTRAINED_MOVE, through + // org.eclipse.gef.requests.ChangeBoundsRequest.isConstrainedMove(), is not handle in this mode. Indeed, the + // user only moves the node in one direction. + // Set the snap deactivation if the modifier key is pressed. + req.setSnapToEnabled(!getCurrentInput().isModKeyDown(MODIFIER_NO_SNAPPING)); + // Set target edit part as parent: The parent remains the same. + setTargetEditPart(getSourceEditPart().getParent()); + req.setType(getCommandName()); + req.setEditParts(getOperationSet()); + + setState(STATE_ACCESSIBLE_DRAG_IN_PROGRESS); + handleDragInProgress(); } - return super.handleKeyDown(event); + return result; } + /** + * Method overridden to handle snapToAll and to disable the new mode in case of move with arrow key. This mode has + * been enabled in {@link #handleKeyDown(KeyEvent)}. + * + * @see org.eclipse.sirius.diagram.ui.tools.internal.ui.SnapToAllDragEditPartsTracker#handleKeyUp(org.eclipse.swt.events.KeyEvent) + */ @Override protected boolean handleKeyUp(KeyEvent event) { if (SnapToAllDragEditPartsTracker.SNAP_TO_ALL == event.keyCode) { snapToAllShape = SnapToAllDragEditPartsTracker.DEFAULT_SNAP_TO_SHAPE_MODE; return true; } + if (acceptArrowKey(event)) { + moveWithArrowKeysSiriusMode = false; + siriusAccStepReset(); + } return super.handleKeyUp(event); } @@ -153,4 +329,135 @@ protected boolean handleDragInProgress() { } return super.handleDragInProgress(); } + + /** + * Copied from {@link org.eclipse.gef.tools.AbstractTool#isCurrentViewerMirrored()}. + * + * @return true if the current viewer is mirrored, false otherwise. + */ + private boolean isCurrentViewerMirrored2() { + return (getCurrentViewer().getControl().getStyle() & SWT.MIRRORED) != 0; + } + + /** + * Method overridden to handle the new mode. Only snap is called here because all other steps are not needed in the + * new mode. + * + * @see org.eclipse.gef.tools.DragEditPartsTracker#updateTargetRequest() + */ + @Override + protected void updateTargetRequest() { + if (moveWithArrowKeysSiriusMode) { + snapPoint((ChangeBoundsRequest) getTargetRequest()); + } else { + super.updateTargetRequest(); + } + } + + /** + * Method overridden to do nothing in case of the new mode. Indeed, when a node is move with an arrow key there is + * no target change. + * + * @see org.eclipse.gef.tools.TargetingTool#updateTargetUnderMouse() + */ + @Override + protected boolean updateTargetUnderMouse() { + if (moveWithArrowKeysSiriusMode) { + return false; + } + return super.updateTargetUnderMouse(); + } + + /** + * Copied from org.eclipse.gef.tools.DragEditPartsTracker.captureSourceDimensions(), but using "real absolute + * coordinates", ie without parents scroll.
    + * Captures the bounds of the source being dragged, and the unioned bounds of all figures being dragged. These + * bounds are used for snapping by the snap strategies in updateTargetRequest(). + */ + private PrecisionRectangle[] captureSourceDimensionsWithAbsoluteBoundsAndWithoutScroll() { + PrecisionRectangle sourceRectangle = null; + PrecisionRectangle compoundSrcRect = null; + List editparts = getOperationSet(); + for (int i = 0; i < editparts.size(); i++) { + GraphicalEditPart child = (GraphicalEditPart) editparts.get(i); + PrecisionRectangle bounds = getAbsoluteBoundsWithoutScroll(child); + + if (compoundSrcRect == null) { + compoundSrcRect = new PrecisionRectangle(bounds); + } else { + compoundSrcRect = compoundSrcRect.union(bounds); + } + if (child == getSourceEditPart()) { + sourceRectangle = bounds; + } + } + if (sourceRectangle == null) { + sourceRectangle = getAbsoluteBoundsWithoutScroll((GraphicalEditPart) getSourceEditPart()); + } + return new PrecisionRectangle[] { sourceRectangle, compoundSrcRect }; + } + + private PrecisionRectangle getAbsoluteBoundsWithoutScroll(GraphicalEditPart editPart) { + IFigure figure = editPart.getFigure(); + PrecisionRectangle bounds = null; + if (figure instanceof HandleBounds) { + bounds = new PrecisionRectangle(((HandleBounds) figure).getHandleBounds()); + } else { + bounds = new PrecisionRectangle(figure.getBounds()); + } + // The six lines above replace the old code "figure.translateToAbsolute(bounds);". + Point topLeft = bounds.getTopLeft(); + FigureUtilities.translateToAbsoluteByIgnoringScrollbar(figure, topLeft); + bounds.setX(topLeft.x); + bounds.setY(topLeft.y); + return bounds; + } + + /** + * Method overridden to initialize the cloned field {@link #accessibleBegin}. No longer necessary as soon as GEF + * issue https://github.com/eclipse/gef-classic/issues/426 will be done. + * + * @see org.eclipse.gef.tools.AbstractTool#activate() + */ + @Override + public void activate() { + super.activate(); + siriusAccStepReset(); + } + + /** + * Method cloned from {@link org.eclipse.gef.tools.AbstractTool#.accGetStep()} to have the same behavior here. No + * longer necessary as soon as GEF issue https://github.com/eclipse/gef-classic/issues/426 will be done. + * + * @return the current computed step. + */ + int siriusAccGetStep() { + return accessibleStep; + } + + /** + * Method cloned from {@link org.eclipse.gef.tools.AbstractTool#accStepIncrement()} to have the same behavior here. + * No longer necessary as soon as GEF issue https://github.com/eclipse/gef-classic/issues/426 will be done. + * + * @return the current computed step. + */ + void siriusAccStepIncrement() { + if (accessibleBegin == -1) { + accessibleBegin = new Date().getTime(); + accessibleStep = 1; + } else { + accessibleStep = 4; + long elapsed = new Date().getTime() - accessibleBegin; + if (elapsed > 1000) + accessibleStep = Math.min(16, (int) (elapsed / 150)); + } + } + + /** + * Method cloned from {@link org.eclipse.gef.tools.AbstractTool#accStepReset()} to have the same behavior here. No + * longer necessary as soon as GEF issue https://github.com/eclipse/gef-classic/issues/426 will be done. + */ + void siriusAccStepReset() { + accessibleBegin = -1; + } }