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:
@@ -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,
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -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).
|
||||||
|
|||||||
Reference in New Issue
Block a user