From d7056e91f0bd00bc38c179fc99b7ad52390ab27f 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 | 118 +++++++- .../ui/SnapToAllDragEditPartsTracker.java | 286 +++++++++++++++++- 2 files changed, 394 insertions(+), 10 deletions(-) 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..87d7cd120f 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,24 @@ *******************************************************************************/ 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.sirius.diagram.ui.tools.internal.ui.SnapToAllDragEditPartsTracker; +import org.eclipse.swt.events.KeyEvent; /** * Specific Sirius SelectionToolEx to use findMouseEventTargetAt instead of findObjectAtExcluding. This allows to @@ -76,4 +86,110 @@ 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); + } else { + 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; + } + + @Override + protected boolean handleViewerExited() { + boolean doNothing = false; + if (isInState(STATE_ACCESSIBLE_DRAG | STATE_ACCESSIBLE_DRAG_IN_PROGRESS | STATE_TRAVERSE_HANDLE | STATE_DRAG | STATE_DRAG_IN_PROGRESS)) { + if (getDragTracker() instanceof SnapToAllDragEditPartsTracker && ((SnapToAllDragEditPartsTracker) getDragTracker()).isMoveWithArrowKeysSiriusMode()) { + // We do nothing as the key continue to be pressed + doNothing = true; + } + } + if (doNothing) { + return false; + } else { + return super.handleViewerExited(); + } + } } 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..2b0eeeaefa 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,24 @@ *******************************************************************************/ package org.eclipse.sirius.diagram.ui.tools.internal.ui; +import java.util.Date; + +import org.eclipse.core.runtime.Platform; import org.eclipse.draw2d.FigureCanvas; +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.SharedCursors; 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.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 +44,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 +58,40 @@ public class SnapToAllDragEditPartsTracker extends DragEditPartsTrackerEx implem */ public static final int SNAP_TO_ALL = SWT.F4; + /** + * 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 +116,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 +128,145 @@ 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 { + // 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); + } + // Copied from org.eclipse.gef.tools.DragEditPartsTracker.snapPoint(ChangeBoundsRequest) + if (getSnapToHelper() != null && request.isSnapToEnabled()) { + PrecisionRectangle baseRect = getSourceBounds().getPreciseCopy(); + PrecisionRectangle jointRect = getOperationSetBounds().getPreciseCopy(); + PrecisionPoint preciseDelta = new PrecisionPoint(moveDelta); + baseRect.translate(preciseDelta); + jointRect.translate(preciseDelta); + getSnapToHelper().snapPoint(request, PositionConstants.HORIZONTAL | PositionConstants.VERTICAL, new PrecisionRectangle[] { baseRect, jointRect }, preciseDelta); + request.setMoveDelta(preciseDelta); + } + } } + /** + * 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(); + PrecisionPoint moveDelta; + switch (event.keyCode) { + case SWT.ARROW_DOWN: + moveDelta = new PrecisionPoint(0, siriusAccGetStep()); + break; + case SWT.ARROW_UP: + moveDelta = new PrecisionPoint(0, -siriusAccGetStep()); + break; + case SWT.ARROW_RIGHT: + int stepping = siriusAccGetStep(); + if (isCurrentViewerMirrored2()) { + stepping = -stepping; + } + moveDelta = new PrecisionPoint(stepping, 0); + break; + case SWT.ARROW_LEFT: + int step = -siriusAccGetStep(); + if (isCurrentViewerMirrored2()) { + step = -step; + } + moveDelta = new PrecisionPoint(step, 0); + break; + default: + moveDelta = new PrecisionPoint(0, 0); + } + // Adapt the moveDelta to the zoom level, 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% + double zoomFactor = GraphicalHelper.getZoom(getSourceEditPart()); + PrecisionPoint preciseDelta = (PrecisionPoint) new PrecisionPoint(moveDelta).scale(zoomFactor); + // Set the request according to the move + ChangeBoundsRequest req = (ChangeBoundsRequest) getTargetRequest(); + if (req.getMoveDelta().preciseX() == 0 && req.getMoveDelta().preciseY() == 0) { + req.setMoveDelta(preciseDelta); + } else { + req.setMoveDelta(req.getMoveDelta().getTranslated(preciseDelta)); + } + 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 +316,109 @@ 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(); + } + + /** + * Return true if a move is currently in progress with arrow key, false otherwise. + * + * @return true if a move is currently in progress with arrow key, false otherwise. + */ + public boolean isMoveWithArrowKeysSiriusMode() { + return moveWithArrowKeysSiriusMode; + } + + /** + * Set the moveWithArrowKeysSiriusMode. + * + * @param moveWithArrowKeysSiriusMode + * The new mode status + */ + protected void setMoveWithArrowKeysSiriusMode(boolean moveWithArrowKeysSiriusMode) { + this.moveWithArrowKeysSiriusMode = moveWithArrowKeysSiriusMode; + } + + /** + * 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; + } }