feat: implement edge splitting at existing nodes with position updates

- Added a new mutation to split edges at existing nodes, allowing for dynamic edge management during canvas interactions.
- Enhanced the logic to update node positions in a single transaction, improving user experience during edge manipulations.
- Refactored edge handling to support optimistic updates, ensuring smoother UI feedback during edge splits.
- Introduced error handling for edge and node validation to maintain data integrity during operations.
This commit is contained in:
Matthias
2026-03-28 13:31:41 +01:00
parent fb24205da0
commit b3a1ed54db
2 changed files with 295 additions and 115 deletions

View File

@@ -77,6 +77,8 @@ type PendingEdgeSplit = {
intersectedTargetHandle?: string; intersectedTargetHandle?: string;
middleSourceHandle?: string; middleSourceHandle?: string;
middleTargetHandle?: string; middleTargetHandle?: string;
positionX: number;
positionY: number;
}; };
function withResolvedCompareData(nodes: RFNode[], edges: RFEdge[]): RFNode[] { function withResolvedCompareData(nodes: RFNode[], edges: RFEdge[]): RFNode[] {
@@ -555,87 +557,114 @@ function CanvasInner({ canvasId }: CanvasInnerProps) {
}, },
); );
const commitEdgeIntersectionSplit = useCallback( const splitEdgeAtExistingNodeMut = useMutation(
async ( api.nodes.splitEdgeAtExistingNode,
middleNodeId: Id<"nodes">, ).withOptimisticUpdate((localStore, args) => {
intersectedEdge: RFEdge, const edgeList = localStore.getQuery(api.edges.list, {
handles: NonNullable<(typeof NODE_HANDLE_MAP)[string]>, canvasId: args.canvasId,
) => {
await Promise.all([
createEdge({
canvasId,
sourceNodeId: intersectedEdge.source as Id<"nodes">,
targetNodeId: middleNodeId,
sourceHandle: normalizeHandle(intersectedEdge.sourceHandle),
targetHandle: normalizeHandle(handles.target),
}),
createEdge({
canvasId,
sourceNodeId: middleNodeId,
targetNodeId: intersectedEdge.target as Id<"nodes">,
sourceHandle: normalizeHandle(handles.source),
targetHandle: normalizeHandle(intersectedEdge.targetHandle),
}),
removeEdge({ edgeId: intersectedEdge.id as Id<"edges"> }),
]);
},
[canvasId, createEdge, removeEdge],
);
const flushPendingEdgeSplit = useCallback(
(clientRequestId: string, realMiddleNodeId: Id<"nodes">) => {
const pending = pendingEdgeSplitByClientRequestRef.current.get(
clientRequestId,
);
if (!pending) return;
pendingEdgeSplitByClientRequestRef.current.delete(clientRequestId);
void Promise.all([
createEdge({
canvasId,
sourceNodeId: pending.sourceNodeId,
targetNodeId: realMiddleNodeId,
sourceHandle: pending.intersectedSourceHandle,
targetHandle: pending.middleTargetHandle,
}),
createEdge({
canvasId,
sourceNodeId: realMiddleNodeId,
targetNodeId: pending.targetNodeId,
sourceHandle: pending.middleSourceHandle,
targetHandle: pending.intersectedTargetHandle,
}),
removeEdge({ edgeId: pending.intersectedEdgeId }),
]).catch((error: unknown) => {
console.error("[Canvas pending edge split failed]", {
clientRequestId,
realMiddleNodeId,
error: String(error),
}); });
const nodeList = localStore.getQuery(api.nodes.list, {
canvasId: args.canvasId,
}); });
}, if (edgeList === undefined || nodeList === undefined) return;
[canvasId, createEdge, removeEdge],
);
/** Pairing: create kann vor oder nach Drag-Ende fertig sein — was zuerst kommt, speichert; das andere triggert moveNode. Zusätzlich: Kanten-Split erst mit echter Node-ID (nach create). */ const removed = edgeList.find((e) => e._id === args.splitEdgeId);
if (!removed) return;
const t1 = `${OPTIMISTIC_EDGE_PREFIX}s1_${Date.now()}_${Math.random().toString(36).slice(2, 10)}` as Id<"edges">;
const t2 = `${OPTIMISTIC_EDGE_PREFIX}s2_${Date.now()}_${Math.random().toString(36).slice(2, 10)}` as Id<"edges">;
const now = Date.now();
const nextEdges = edgeList.filter((e) => e._id !== args.splitEdgeId);
nextEdges.push(
{
_id: t1,
_creationTime: now,
canvasId: args.canvasId,
sourceNodeId: removed.sourceNodeId,
targetNodeId: args.middleNodeId,
sourceHandle: args.splitSourceHandle,
targetHandle: args.newNodeTargetHandle,
},
{
_id: t2,
_creationTime: now,
canvasId: args.canvasId,
sourceNodeId: args.middleNodeId,
targetNodeId: removed.targetNodeId,
sourceHandle: args.newNodeSourceHandle,
targetHandle: args.splitTargetHandle,
},
);
localStore.setQuery(api.edges.list, { canvasId: args.canvasId }, nextEdges);
if (args.positionX !== undefined && args.positionY !== undefined) {
const px = args.positionX;
const py = args.positionY;
localStore.setQuery(
api.nodes.list,
{ canvasId: args.canvasId },
nodeList.map((n) =>
n._id === args.middleNodeId
? {
...n,
positionX: px,
positionY: py,
}
: n,
),
);
}
});
/** Pairing: create kann vor oder nach Drag-Ende fertig sein. Kanten-Split + Position in einem Convex-Roundtrip wenn split ansteht. */
const syncPendingMoveForClientRequest = useCallback( const syncPendingMoveForClientRequest = useCallback(
(clientRequestId: string | undefined, realId?: Id<"nodes">) => { (clientRequestId: string | undefined, realId?: Id<"nodes">) => {
if (!clientRequestId) return; if (!clientRequestId) return;
if (realId !== undefined) { if (realId !== undefined) {
const pending = pendingMoveAfterCreateRef.current.get(clientRequestId); const pendingMove = pendingMoveAfterCreateRef.current.get(clientRequestId);
if (pending) { const splitPayload =
pendingEdgeSplitByClientRequestRef.current.get(clientRequestId);
if (splitPayload) {
pendingEdgeSplitByClientRequestRef.current.delete(clientRequestId);
if (pendingMove) {
pendingMoveAfterCreateRef.current.delete(clientRequestId);
}
resolvedRealIdByClientRequestRef.current.delete(clientRequestId);
void splitEdgeAtExistingNodeMut({
canvasId,
splitEdgeId: splitPayload.intersectedEdgeId,
middleNodeId: realId,
splitSourceHandle: splitPayload.intersectedSourceHandle,
splitTargetHandle: splitPayload.intersectedTargetHandle,
newNodeSourceHandle: splitPayload.middleSourceHandle,
newNodeTargetHandle: splitPayload.middleTargetHandle,
positionX: pendingMove?.positionX ?? splitPayload.positionX,
positionY: pendingMove?.positionY ?? splitPayload.positionY,
}).catch((error: unknown) => {
console.error("[Canvas pending edge split failed]", {
clientRequestId,
realId,
error: String(error),
});
});
return;
}
if (pendingMove) {
pendingMoveAfterCreateRef.current.delete(clientRequestId); pendingMoveAfterCreateRef.current.delete(clientRequestId);
resolvedRealIdByClientRequestRef.current.delete(clientRequestId); resolvedRealIdByClientRequestRef.current.delete(clientRequestId);
void moveNode({ void moveNode({
nodeId: realId, nodeId: realId,
positionX: pending.positionX, positionX: pendingMove.positionX,
positionY: pending.positionY, positionY: pendingMove.positionY,
}); });
flushPendingEdgeSplit(clientRequestId, realId);
return; return;
} }
resolvedRealIdByClientRequestRef.current.set(clientRequestId, realId); resolvedRealIdByClientRequestRef.current.set(clientRequestId, realId);
flushPendingEdgeSplit(clientRequestId, realId);
return; return;
} }
@@ -644,14 +673,37 @@ function CanvasInner({ canvasId }: CanvasInnerProps) {
if (!r || !p) return; if (!r || !p) return;
pendingMoveAfterCreateRef.current.delete(clientRequestId); pendingMoveAfterCreateRef.current.delete(clientRequestId);
resolvedRealIdByClientRequestRef.current.delete(clientRequestId); resolvedRealIdByClientRequestRef.current.delete(clientRequestId);
const splitPayload =
pendingEdgeSplitByClientRequestRef.current.get(clientRequestId);
if (splitPayload) {
pendingEdgeSplitByClientRequestRef.current.delete(clientRequestId);
void splitEdgeAtExistingNodeMut({
canvasId,
splitEdgeId: splitPayload.intersectedEdgeId,
middleNodeId: r,
splitSourceHandle: splitPayload.intersectedSourceHandle,
splitTargetHandle: splitPayload.intersectedTargetHandle,
newNodeSourceHandle: splitPayload.middleSourceHandle,
newNodeTargetHandle: splitPayload.middleTargetHandle,
positionX: splitPayload.positionX ?? p.positionX,
positionY: splitPayload.positionY ?? p.positionY,
}).catch((error: unknown) => {
console.error("[Canvas pending edge split failed]", {
clientRequestId,
realId: r,
error: String(error),
});
});
} else {
void moveNode({ void moveNode({
nodeId: r, nodeId: r,
positionX: p.positionX, positionX: p.positionX,
positionY: p.positionY, positionY: p.positionY,
}); });
flushPendingEdgeSplit(clientRequestId, r); }
}, },
[moveNode, flushPendingEdgeSplit], [canvasId, moveNode, splitEdgeAtExistingNodeMut],
); );
// ─── Lokaler State (für flüssiges Dragging) ─────────────────── // ─── Lokaler State (für flüssiges Dragging) ───────────────────
@@ -1152,7 +1204,22 @@ function CanvasInner({ canvasId }: CanvasInnerProps) {
void (async () => { void (async () => {
try { try {
// isDragging bleibt true bis alle Mutations resolved sind const intersectedEdge = intersectedEdgeId
? edges.find(
(edge) =>
edge.id === intersectedEdgeId && edge.className !== "temp",
)
: undefined;
const splitHandles = NODE_HANDLE_MAP[node.type ?? ""];
const splitEligible =
intersectedEdge !== undefined &&
splitHandles !== undefined &&
intersectedEdge.source !== node.id &&
intersectedEdge.target !== node.id &&
hasHandleKey(splitHandles, "source") &&
hasHandleKey(splitHandles, "target");
if (draggedNodes.length > 1) { if (draggedNodes.length > 1) {
for (const n of draggedNodes) { for (const n of draggedNodes) {
const cid = clientRequestIdFromOptimisticNodeId(n.id); const cid = clientRequestIdFromOptimisticNodeId(n.id);
@@ -1174,53 +1241,17 @@ function CanvasInner({ canvasId }: CanvasInnerProps) {
})), })),
}); });
} }
} else {
const cid = clientRequestIdFromOptimisticNodeId(node.id);
if (cid) {
pendingMoveAfterCreateRef.current.set(cid, {
positionX: node.position.x,
positionY: node.position.y,
});
syncPendingMoveForClientRequest(cid);
} else {
await moveNode({
nodeId: node.id as Id<"nodes">,
positionX: node.position.x,
positionY: node.position.y,
});
}
}
if (!intersectedEdgeId) { if (!splitEligible || !intersectedEdge) {
return; return;
} }
const intersectedEdge = edges.find((edge) => edge.id === intersectedEdgeId); const multiCid = clientRequestIdFromOptimisticNodeId(node.id);
if (!intersectedEdge || intersectedEdge.className === "temp") { let middleId = node.id as Id<"nodes">;
return; if (multiCid) {
} const r = resolvedRealIdByClientRequestRef.current.get(multiCid);
if (!r) {
if ( pendingEdgeSplitByClientRequestRef.current.set(multiCid, {
intersectedEdge.source === node.id ||
intersectedEdge.target === node.id
) {
return;
}
const handles = NODE_HANDLE_MAP[node.type ?? ""];
if (!hasHandleKey(handles, "source") || !hasHandleKey(handles, "target")) {
return;
}
const optimisticCid = clientRequestIdFromOptimisticNodeId(node.id);
let middleNodeId = node.id as Id<"nodes">;
if (optimisticCid) {
const resolvedMiddle =
resolvedRealIdByClientRequestRef.current.get(optimisticCid);
if (resolvedMiddle) {
middleNodeId = resolvedMiddle;
} else {
pendingEdgeSplitByClientRequestRef.current.set(optimisticCid, {
intersectedEdgeId: intersectedEdge.id as Id<"edges">, intersectedEdgeId: intersectedEdge.id as Id<"edges">,
sourceNodeId: intersectedEdge.source as Id<"nodes">, sourceNodeId: intersectedEdge.source as Id<"nodes">,
targetNodeId: intersectedEdge.target as Id<"nodes">, targetNodeId: intersectedEdge.target as Id<"nodes">,
@@ -1230,18 +1261,99 @@ function CanvasInner({ canvasId }: CanvasInnerProps) {
intersectedTargetHandle: normalizeHandle( intersectedTargetHandle: normalizeHandle(
intersectedEdge.targetHandle, intersectedEdge.targetHandle,
), ),
middleSourceHandle: normalizeHandle(handles.source), middleSourceHandle: normalizeHandle(splitHandles.source),
middleTargetHandle: normalizeHandle(handles.target), middleTargetHandle: normalizeHandle(splitHandles.target),
positionX: node.position.x,
positionY: node.position.y,
}); });
return; return;
} }
middleId = r;
} }
await commitEdgeIntersectionSplit( await splitEdgeAtExistingNodeMut({
middleNodeId, canvasId,
intersectedEdge, splitEdgeId: intersectedEdge.id as Id<"edges">,
handles, middleNodeId: middleId,
); splitSourceHandle: normalizeHandle(intersectedEdge.sourceHandle),
splitTargetHandle: normalizeHandle(intersectedEdge.targetHandle),
newNodeSourceHandle: normalizeHandle(splitHandles.source),
newNodeTargetHandle: normalizeHandle(splitHandles.target),
});
return;
}
if (!splitEligible || !intersectedEdge) {
const cidSingle = clientRequestIdFromOptimisticNodeId(node.id);
if (cidSingle) {
pendingMoveAfterCreateRef.current.set(cidSingle, {
positionX: node.position.x,
positionY: node.position.y,
});
syncPendingMoveForClientRequest(cidSingle);
} else {
await moveNode({
nodeId: node.id as Id<"nodes">,
positionX: node.position.x,
positionY: node.position.y,
});
}
return;
}
const singleCid = clientRequestIdFromOptimisticNodeId(node.id);
if (singleCid) {
const resolvedSingle =
resolvedRealIdByClientRequestRef.current.get(singleCid);
if (!resolvedSingle) {
pendingMoveAfterCreateRef.current.set(singleCid, {
positionX: node.position.x,
positionY: node.position.y,
});
pendingEdgeSplitByClientRequestRef.current.set(singleCid, {
intersectedEdgeId: intersectedEdge.id as Id<"edges">,
sourceNodeId: intersectedEdge.source as Id<"nodes">,
targetNodeId: intersectedEdge.target as Id<"nodes">,
intersectedSourceHandle: normalizeHandle(
intersectedEdge.sourceHandle,
),
intersectedTargetHandle: normalizeHandle(
intersectedEdge.targetHandle,
),
middleSourceHandle: normalizeHandle(splitHandles.source),
middleTargetHandle: normalizeHandle(splitHandles.target),
positionX: node.position.x,
positionY: node.position.y,
});
syncPendingMoveForClientRequest(singleCid);
return;
}
await splitEdgeAtExistingNodeMut({
canvasId,
splitEdgeId: intersectedEdge.id as Id<"edges">,
middleNodeId: resolvedSingle,
splitSourceHandle: normalizeHandle(intersectedEdge.sourceHandle),
splitTargetHandle: normalizeHandle(intersectedEdge.targetHandle),
newNodeSourceHandle: normalizeHandle(splitHandles.source),
newNodeTargetHandle: normalizeHandle(splitHandles.target),
positionX: node.position.x,
positionY: node.position.y,
});
pendingMoveAfterCreateRef.current.delete(singleCid);
return;
}
await splitEdgeAtExistingNodeMut({
canvasId,
splitEdgeId: intersectedEdge.id as Id<"edges">,
middleNodeId: node.id as Id<"nodes">,
splitSourceHandle: normalizeHandle(intersectedEdge.sourceHandle),
splitTargetHandle: normalizeHandle(intersectedEdge.targetHandle),
newNodeSourceHandle: normalizeHandle(splitHandles.source),
newNodeTargetHandle: normalizeHandle(splitHandles.target),
positionX: node.position.x,
positionY: node.position.y,
});
} catch (error) { } catch (error) {
console.error("[Canvas edge intersection split failed]", { console.error("[Canvas edge intersection split failed]", {
canvasId, canvasId,
@@ -1260,10 +1372,10 @@ function CanvasInner({ canvasId }: CanvasInnerProps) {
[ [
batchMoveNodes, batchMoveNodes,
canvasId, canvasId,
commitEdgeIntersectionSplit,
edges, edges,
moveNode, moveNode,
setHighlightedIntersectionEdge, setHighlightedIntersectionEdge,
splitEdgeAtExistingNodeMut,
syncPendingMoveForClientRequest, syncPendingMoveForClientRequest,
], ],
); );

View File

@@ -224,6 +224,74 @@ export const createWithEdgeSplit = mutation({
}, },
}); });
/**
* Bestehenden Knoten in eine Kante einhängen: alte Kante löschen, zwei neue anlegen.
* Optional positionX/Y: Mitte-Knoten in derselben Transaktion verschieben (ein Roundtrip mit Drag-Ende).
*/
export const splitEdgeAtExistingNode = mutation({
args: {
canvasId: v.id("canvases"),
splitEdgeId: v.id("edges"),
middleNodeId: v.id("nodes"),
splitSourceHandle: v.optional(v.string()),
splitTargetHandle: v.optional(v.string()),
newNodeSourceHandle: v.optional(v.string()),
newNodeTargetHandle: v.optional(v.string()),
positionX: v.optional(v.number()),
positionY: v.optional(v.number()),
},
handler: async (ctx, args) => {
const user = await requireAuth(ctx);
await getCanvasOrThrow(ctx, args.canvasId, user.userId);
const edge = await ctx.db.get(args.splitEdgeId);
if (!edge || edge.canvasId !== args.canvasId) {
throw new Error("Edge not found");
}
if (
edge.sourceNodeId === args.middleNodeId ||
edge.targetNodeId === args.middleNodeId
) {
throw new Error("Middle node is already an endpoint of this edge");
}
const middle = await ctx.db.get(args.middleNodeId);
if (!middle || middle.canvasId !== args.canvasId) {
throw new Error("Middle node not found");
}
if (
args.positionX !== undefined &&
args.positionY !== undefined
) {
await ctx.db.patch(args.middleNodeId, {
positionX: args.positionX,
positionY: args.positionY,
});
}
await ctx.db.insert("edges", {
canvasId: args.canvasId,
sourceNodeId: edge.sourceNodeId,
targetNodeId: args.middleNodeId,
sourceHandle: args.splitSourceHandle,
targetHandle: args.newNodeTargetHandle,
});
await ctx.db.insert("edges", {
canvasId: args.canvasId,
sourceNodeId: args.middleNodeId,
targetNodeId: edge.targetNodeId,
sourceHandle: args.newNodeSourceHandle,
targetHandle: args.splitTargetHandle,
});
await ctx.db.delete(args.splitEdgeId);
await ctx.db.patch(args.canvasId, { updatedAt: Date.now() });
},
});
/** /**
* Neuen Node erstellen und sofort mit einem bestehenden Node verbinden * Neuen Node erstellen und sofort mit einem bestehenden Node verbinden
* (ein Roundtrip — z. B. Prompt → neue AI-Image-Node). * (ein Roundtrip — z. B. Prompt → neue AI-Image-Node).