From 0ab2809ce9c322a4456633689d631db000179401 Mon Sep 17 00:00:00 2001 From: Beka Westberg Date: Fri, 8 Mar 2024 16:45:11 +0000 Subject: [PATCH] fix: connection dancing (#7899) * fix: special case zelos for connection previewing * chore: add TODO (cherry picked from commit 848415e48f63c4ac2cd6ae2d77a6ecdba6cec56a) --- .../insertion_marker_previewer.ts | 89 ++++++++++++------- 1 file changed, 56 insertions(+), 33 deletions(-) diff --git a/core/connection_previewers/insertion_marker_previewer.ts b/core/connection_previewers/insertion_marker_previewer.ts index 3f6f1584149..414d8dfe7da 100644 --- a/core/connection_previewers/insertion_marker_previewer.ts +++ b/core/connection_previewers/insertion_marker_previewer.ts @@ -12,6 +12,8 @@ import * as eventUtils from '../events/utils.js'; import * as constants from '../constants.js'; import * as renderManagement from '../render_management.js'; import * as registry from '../registry.js'; +import {Renderer as ZelosRenderer} from '../renderers/zelos/renderer.js'; +import {ConnectionType} from '../connection_type.js'; /** * An error message to throw if the block created by createMarkerBlock_ is @@ -86,45 +88,19 @@ export class InsertionMarkerPreviewer implements IConnectionPreviewer { eventUtils.disable(); try { this.hidePreview(); - const dragged = draggedConn.getSourceBlock(); - const marker = this.createInsertionMarker(dragged); - const markerConn = this.getMatchingConnection( - dragged, - marker, - draggedConn, - ); - if (!markerConn) { - throw Error('Could not create insertion marker to preview connection'); - } - - // Render disconnected from everything else so that we have a valid - // connection location. - marker.queueRender(); - renderManagement.triggerQueuedRenders(); - // Connect() also renders the insertion marker. - markerConn.connect(staticConn); - - const originalOffsetToTarget = { - x: staticConn.x - markerConn.x, - y: staticConn.y - markerConn.y, - }; - const originalOffsetInBlock = markerConn.getOffsetInBlock().clone(); - renderManagement.finishQueuedRenders().then(() => { - // Position so that the existing block doesn't move. - marker?.positionNearConnection( - markerConn, - originalOffsetToTarget, - originalOffsetInBlock, - ); - marker?.getSvgRoot().setAttribute('visibility', 'visible'); - }); + // TODO(7898): Instead of special casing, we should change the dragger to + // track the change in distance between the dragged connection and the + // static connection, so that it doesn't disconnect unless that + // (+ a bit) has been exceeded. + if (this.shouldUseMarkerPreview(draggedConn, staticConn)) { + this.markerConn = this.previewMarker(draggedConn, staticConn); + } if (this.workspace.getRenderer().shouldHighlightConnection(staticConn)) { staticConn.highlight(); } - this.markerConn = markerConn; this.draggedConn = draggedConn; this.staticConn = staticConn; } finally { @@ -132,6 +108,53 @@ export class InsertionMarkerPreviewer implements IConnectionPreviewer { } } + private shouldUseMarkerPreview( + _draggedConn: RenderedConnection, + staticConn: RenderedConnection, + ): boolean { + return ( + staticConn.type === ConnectionType.PREVIOUS_STATEMENT || + staticConn.type === ConnectionType.NEXT_STATEMENT || + !(this.workspace.getRenderer() instanceof ZelosRenderer) + ); + } + + private previewMarker( + draggedConn: RenderedConnection, + staticConn: RenderedConnection, + ): RenderedConnection { + const dragged = draggedConn.getSourceBlock(); + const marker = this.createInsertionMarker(dragged); + const markerConn = this.getMatchingConnection(dragged, marker, draggedConn); + if (!markerConn) { + throw Error('Could not create insertion marker to preview connection'); + } + + // Render disconnected from everything else so that we have a valid + // connection location. + marker.queueRender(); + renderManagement.triggerQueuedRenders(); + + // Connect() also renders the insertion marker. + markerConn.connect(staticConn); + + const originalOffsetToTarget = { + x: staticConn.x - markerConn.x, + y: staticConn.y - markerConn.y, + }; + const originalOffsetInBlock = markerConn.getOffsetInBlock().clone(); + renderManagement.finishQueuedRenders().then(() => { + // Position so that the existing block doesn't move. + marker?.positionNearConnection( + markerConn, + originalOffsetToTarget, + originalOffsetInBlock, + ); + marker?.getSvgRoot().setAttribute('visibility', 'visible'); + }); + return markerConn; + } + private createInsertionMarker(origBlock: BlockSvg) { const result = this.workspace.newBlock(origBlock.type); result.setInsertionMarker(true);