Skip to content

Commit 876b71d

Browse files
committed
refactor(core): use node/edge id as only dependency for nodes/edges list (#1447)
* refactor(core): use node id to (re-)render nodes * refactor(core): use edge id to (re-)render nodes * chore(changeset): add * chore(core): cleanup * chore(core): cast cmp as any
1 parent 8bc1c99 commit 876b71d

File tree

7 files changed

+273
-324
lines changed

7 files changed

+273
-324
lines changed

.changeset/calm-pigs-yawn.md

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@vue-flow/core": minor
3+
---
4+
5+
Return non-nullable edge from `useEdge`

.changeset/four-singers-suffer.md

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"@vue-flow/core": minor
3+
---
4+
5+
Use node/edge id as the only dependency to render nodes/edges.

packages/core/src/components/Edges/EdgeWrapper.ts

Lines changed: 110 additions & 79 deletions
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,8 @@
1-
import { computed, defineComponent, h, provide, ref } from 'vue'
2-
import { useVModel } from '@vueuse/core'
3-
import type { Connection, EdgeComponent, EdgeUpdatable, GraphEdge, HandleType, MouseTouchEvent } from '../../types'
1+
import { computed, defineComponent, getCurrentInstance, h, inject, provide, ref, resolveComponent, toRef } from 'vue'
2+
import type { Connection, EdgeComponent, HandleType, MouseTouchEvent } from '../../types'
43
import { ConnectionMode, Position } from '../../types'
5-
import { useEdgeHooks, useHandle, useVueFlow } from '../../composables'
6-
import { EdgeId, EdgeRef } from '../../context'
4+
import { useEdge, useEdgeHooks, useHandle, useVueFlow } from '../../composables'
5+
import { EdgeId, EdgeRef, Slots } from '../../context'
76
import {
87
ARIA_EDGE_DESC_KEY,
98
ErrorCode,
@@ -17,18 +16,12 @@ import EdgeAnchor from './EdgeAnchor'
1716

1817
interface Props {
1918
id: string
20-
type: EdgeComponent | ((...args: any[]) => any) | object | false
21-
name: string
22-
selectable?: boolean
23-
focusable?: boolean
24-
updatable?: EdgeUpdatable
25-
edge: GraphEdge
2619
}
2720

2821
const EdgeWrapper = defineComponent({
2922
name: 'Edge',
3023
compatConfig: { MODE: 3 },
31-
props: ['name', 'type', 'id', 'updatable', 'selectable', 'focusable', 'edge'],
24+
props: ['id'],
3225
setup(props: Props) {
3326
const {
3427
id: vueFlowId,
@@ -45,11 +38,18 @@ const EdgeWrapper = defineComponent({
4538
isValidConnection,
4639
multiSelectionActive,
4740
disableKeyboardA11y,
41+
elementsSelectable,
42+
edgesUpdatable,
43+
edgesFocusable,
4844
} = useVueFlow()
4945

50-
const hooks = useEdgeHooks(props.edge, emits)
46+
const { edge } = useEdge(props.id)
5147

52-
const edge = useVModel(props, 'edge')
48+
const hooks = useEdgeHooks(edge, emits)
49+
50+
const slots = inject(Slots)
51+
52+
const instance = getCurrentInstance()
5353

5454
const mouseOver = ref(false)
5555

@@ -63,11 +63,45 @@ const EdgeWrapper = defineComponent({
6363

6464
const edgeEl = ref<SVGElement | null>(null)
6565

66+
const isSelectable = toRef(() => (typeof edge.selectable === 'undefined' ? elementsSelectable.value : edge.selectable))
67+
68+
const isUpdatable = toRef(() => (typeof edge.updatable === 'undefined' ? edgesUpdatable.value : edge.updatable))
69+
70+
const isFocusable = toRef(() => (typeof edge.focusable === 'undefined' ? edgesFocusable.value : edge.focusable))
71+
6672
provide(EdgeId, props.id)
6773
provide(EdgeRef, edgeEl)
6874

69-
const edgeClass = computed(() => (edge.value.class instanceof Function ? edge.value.class(edge.value) : edge.value.class))
70-
const edgeStyle = computed(() => (edge.value.style instanceof Function ? edge.value.style(edge.value) : edge.value.style))
75+
const edgeClass = computed(() => (edge.class instanceof Function ? edge.class(edge) : edge.class))
76+
const edgeStyle = computed(() => (edge.style instanceof Function ? edge.style(edge) : edge.style))
77+
78+
const edgeCmp = computed(() => {
79+
const name = edge.type || 'default'
80+
81+
const slot = slots?.[`edge-${name}`]
82+
if (slot) {
83+
return slot
84+
}
85+
86+
let edgeType = edge.template ?? getEdgeTypes.value[name]
87+
88+
if (typeof edgeType === 'string') {
89+
if (instance) {
90+
const components = Object.keys(instance.appContext.components)
91+
if (components && components.includes(name)) {
92+
edgeType = resolveComponent(name, false) as EdgeComponent
93+
}
94+
}
95+
}
96+
97+
if (edgeType && typeof edgeType !== 'string') {
98+
return edgeType
99+
}
100+
101+
emits.error(new VueFlowError(ErrorCode.EDGE_TYPE_MISSING, edgeType))
102+
103+
return false
104+
})
71105

72106
const { handlePointerDown } = useHandle({
73107
nodeId,
@@ -80,29 +114,29 @@ const EdgeWrapper = defineComponent({
80114
})
81115

82116
return () => {
83-
const sourceNode = findNode(edge.value.source)
84-
const targetNode = findNode(edge.value.target)
85-
const pathOptions = 'pathOptions' in edge.value ? edge.value.pathOptions : {}
117+
const sourceNode = findNode(edge.source)
118+
const targetNode = findNode(edge.target)
119+
const pathOptions = 'pathOptions' in edge ? edge.pathOptions : {}
86120

87121
if (!sourceNode && !targetNode) {
88-
emits.error(new VueFlowError(ErrorCode.EDGE_SOURCE_TARGET_MISSING, edge.value.id, edge.value.source, edge.value.target))
122+
emits.error(new VueFlowError(ErrorCode.EDGE_SOURCE_TARGET_MISSING, edge.id, edge.source, edge.target))
89123

90124
return null
91125
}
92126

93127
if (!sourceNode) {
94-
emits.error(new VueFlowError(ErrorCode.EDGE_SOURCE_MISSING, edge.value.id, edge.value.source))
128+
emits.error(new VueFlowError(ErrorCode.EDGE_SOURCE_MISSING, edge.id, edge.source))
95129

96130
return null
97131
}
98132

99133
if (!targetNode) {
100-
emits.error(new VueFlowError(ErrorCode.EDGE_TARGET_MISSING, edge.value.id, edge.value.target))
134+
emits.error(new VueFlowError(ErrorCode.EDGE_TARGET_MISSING, edge.id, edge.target))
101135

102136
return null
103137
}
104138

105-
if (!edge.value || edge.value.hidden || sourceNode.hidden || targetNode.hidden) {
139+
if (!edge || edge.hidden || sourceNode.hidden || targetNode.hidden) {
106140
return null
107141
}
108142

@@ -113,7 +147,7 @@ const EdgeWrapper = defineComponent({
113147
sourceNodeHandles = [...(sourceNode.handleBounds.source || []), ...(sourceNode.handleBounds.target || [])]
114148
}
115149

116-
const sourceHandle = getHandle(sourceNodeHandles, edge.value.sourceHandle)
150+
const sourceHandle = getHandle(sourceNodeHandles, edge.sourceHandle)
117151

118152
let targetNodeHandles
119153
if (connectionMode.value === ConnectionMode.Strict) {
@@ -122,7 +156,7 @@ const EdgeWrapper = defineComponent({
122156
targetNodeHandles = [...(targetNode.handleBounds.target || []), ...(targetNode.handleBounds.source || [])]
123157
}
124158

125-
const targetHandle = getHandle(targetNodeHandles, edge.value.targetHandle)
159+
const targetHandle = getHandle(targetNodeHandles, edge.targetHandle)
126160

127161
const sourcePosition = sourceHandle ? sourceHandle.position : Position.Bottom
128162

@@ -137,10 +171,10 @@ const EdgeWrapper = defineComponent({
137171
targetPosition,
138172
)
139173

140-
edge.value.sourceX = sourceX
141-
edge.value.sourceY = sourceY
142-
edge.value.targetX = targetX
143-
edge.value.targetY = targetY
174+
edge.sourceX = sourceX
175+
edge.sourceY = sourceY
176+
edge.targetX = targetX
177+
edge.targetY = targetY
144178

145179
return h(
146180
'g',
@@ -150,14 +184,14 @@ const EdgeWrapper = defineComponent({
150184
'data-id': props.id,
151185
'class': [
152186
'vue-flow__edge',
153-
`vue-flow__edge-${props.type === false ? 'default' : props.name}`,
187+
`vue-flow__edge-${edgeCmp.value === false ? 'default' : edge.type || 'default'}`,
154188
noPanClassName.value,
155189
edgeClass.value,
156190
{
157191
updating: mouseOver.value,
158-
selected: edge.value.selected,
159-
animated: edge.value.animated,
160-
inactive: !props.selectable,
192+
selected: edge.selected,
193+
animated: edge.animated,
194+
inactive: !isSelectable.value,
161195
},
162196
],
163197
'onClick': onEdgeClick,
@@ -166,52 +200,49 @@ const EdgeWrapper = defineComponent({
166200
'onMouseenter': onEdgeMouseEnter,
167201
'onMousemove': onEdgeMouseMove,
168202
'onMouseleave': onEdgeMouseLeave,
169-
'onKeyDown': props.focusable ? onKeyDown : undefined,
170-
'tabIndex': props.focusable ? 0 : undefined,
171-
'aria-label':
172-
edge.value.ariaLabel === null
173-
? undefined
174-
: edge.value.ariaLabel || `Edge from ${edge.value.source} to ${edge.value.target}`,
175-
'aria-describedby': props.focusable ? `${ARIA_EDGE_DESC_KEY}-${vueFlowId}` : undefined,
176-
'role': props.focusable ? 'button' : 'img',
203+
'onKeyDown': isFocusable.value ? onKeyDown : undefined,
204+
'tabIndex': isFocusable.value ? 0 : undefined,
205+
'aria-label': edge.ariaLabel === null ? undefined : edge.ariaLabel || `Edge from ${edge.source} to ${edge.target}`,
206+
'aria-describedby': isFocusable.value ? `${ARIA_EDGE_DESC_KEY}-${vueFlowId}` : undefined,
207+
'role': isFocusable.value ? 'button' : 'img',
177208
},
178209
[
179210
updating.value
180211
? null
181-
: h(props.type === false ? getEdgeTypes.value.default : props.type, {
212+
: h(edgeCmp.value === false ? getEdgeTypes.value.default : (edgeCmp.value as any), {
182213
id: props.id,
183214
sourceNode,
184215
targetNode,
185-
source: edge.value.source,
186-
target: edge.value.target,
187-
type: edge.value.type,
188-
updatable: props.updatable,
189-
selected: edge.value.selected,
190-
animated: edge.value.animated,
191-
label: edge.value.label,
192-
labelStyle: edge.value.labelStyle,
193-
labelShowBg: edge.value.labelShowBg,
194-
labelBgStyle: edge.value.labelBgStyle,
195-
labelBgPadding: edge.value.labelBgPadding,
196-
labelBgBorderRadius: edge.value.labelBgBorderRadius,
197-
data: edge.value.data,
198-
events: { ...edge.value.events, ...hooks.on },
216+
source: edge.source,
217+
target: edge.target,
218+
type: edge.type,
219+
updatable: isUpdatable.value,
220+
selected: edge.selected,
221+
animated: edge.animated,
222+
label: edge.label,
223+
labelStyle: edge.labelStyle,
224+
labelShowBg: edge.labelShowBg,
225+
labelBgStyle: edge.labelBgStyle,
226+
labelBgPadding: edge.labelBgPadding,
227+
labelBgBorderRadius: edge.labelBgBorderRadius,
228+
data: edge.data,
229+
events: { ...edge.events, ...hooks.on },
199230
style: edgeStyle.value,
200-
markerStart: `url('#${getMarkerId(edge.value.markerStart, vueFlowId)}')`,
201-
markerEnd: `url('#${getMarkerId(edge.value.markerEnd, vueFlowId)}')`,
231+
markerStart: `url('#${getMarkerId(edge.markerStart, vueFlowId)}')`,
232+
markerEnd: `url('#${getMarkerId(edge.markerEnd, vueFlowId)}')`,
202233
sourcePosition,
203234
targetPosition,
204235
sourceX,
205236
sourceY,
206237
targetX,
207238
targetY,
208-
sourceHandleId: edge.value.sourceHandle,
209-
targetHandleId: edge.value.targetHandle,
210-
interactionWidth: edge.value.interactionWidth,
239+
sourceHandleId: edge.sourceHandle,
240+
targetHandleId: edge.targetHandle,
241+
interactionWidth: edge.interactionWidth,
211242
...pathOptions,
212243
}),
213244
[
214-
props.updatable === 'source' || props.updatable === true
245+
isUpdatable.value === 'source' || isUpdatable.value === true
215246
? [
216247
h(
217248
'g',
@@ -231,7 +262,7 @@ const EdgeWrapper = defineComponent({
231262
),
232263
]
233264
: null,
234-
props.updatable === 'target' || props.updatable === true
265+
isUpdatable.value === 'target' || isUpdatable.value === true
235266
? [
236267
h(
237268
'g',
@@ -265,11 +296,11 @@ const EdgeWrapper = defineComponent({
265296
}
266297

267298
function onEdgeUpdate(event: MouseTouchEvent, connection: Connection) {
268-
hooks.emit.update({ event, edge: edge.value, connection })
299+
hooks.emit.update({ event, edge, connection })
269300
}
270301

271302
function onEdgeUpdateEnd(event: MouseTouchEvent) {
272-
hooks.emit.updateEnd({ event, edge: edge.value })
303+
hooks.emit.updateEnd({ event, edge })
273304
updating.value = false
274305
}
275306

@@ -280,52 +311,52 @@ const EdgeWrapper = defineComponent({
280311

281312
updating.value = true
282313

283-
nodeId.value = isSourceHandle ? edge.value.target : edge.value.source
284-
handleId.value = (isSourceHandle ? edge.value.targetHandle : edge.value.sourceHandle) ?? ''
314+
nodeId.value = isSourceHandle ? edge.target : edge.source
315+
handleId.value = (isSourceHandle ? edge.targetHandle : edge.sourceHandle) ?? ''
285316

286317
edgeUpdaterType.value = isSourceHandle ? 'target' : 'source'
287318

288-
hooks.emit.updateStart({ event, edge: edge.value })
319+
hooks.emit.updateStart({ event, edge })
289320

290321
handlePointerDown(event)
291322
}
292323

293324
function onEdgeClick(event: MouseEvent) {
294-
const data = { event, edge: edge.value }
325+
const data = { event, edge }
295326

296-
if (props.selectable) {
327+
if (isSelectable.value) {
297328
nodesSelectionActive.value = false
298329

299-
if (edge.value.selected && multiSelectionActive.value) {
300-
removeSelectedEdges([edge.value])
330+
if (edge.selected && multiSelectionActive.value) {
331+
removeSelectedEdges([edge])
301332

302333
edgeEl.value?.blur()
303334
} else {
304-
addSelectedEdges([edge.value])
335+
addSelectedEdges([edge])
305336
}
306337
}
307338

308339
hooks.emit.click(data)
309340
}
310341

311342
function onEdgeContextMenu(event: MouseEvent) {
312-
hooks.emit.contextMenu({ event, edge: edge.value })
343+
hooks.emit.contextMenu({ event, edge })
313344
}
314345

315346
function onDoubleClick(event: MouseEvent) {
316-
hooks.emit.doubleClick({ event, edge: edge.value })
347+
hooks.emit.doubleClick({ event, edge })
317348
}
318349

319350
function onEdgeMouseEnter(event: MouseEvent) {
320-
hooks.emit.mouseEnter({ event, edge: edge.value })
351+
hooks.emit.mouseEnter({ event, edge })
321352
}
322353

323354
function onEdgeMouseMove(event: MouseEvent) {
324-
hooks.emit.mouseMove({ event, edge: edge.value })
355+
hooks.emit.mouseMove({ event, edge })
325356
}
326357

327358
function onEdgeMouseLeave(event: MouseEvent) {
328-
hooks.emit.mouseLeave({ event, edge: edge.value })
359+
hooks.emit.mouseLeave({ event, edge })
329360
}
330361

331362
function onEdgeUpdaterSourceMouseDown(event: MouseEvent) {
@@ -337,7 +368,7 @@ const EdgeWrapper = defineComponent({
337368
}
338369

339370
function onKeyDown(event: KeyboardEvent) {
340-
if (!disableKeyboardA11y.value && elementSelectionKeys.includes(event.key) && props.selectable) {
371+
if (!disableKeyboardA11y.value && elementSelectionKeys.includes(event.key) && isSelectable.value) {
341372
const unselect = event.key === 'Escape'
342373

343374
if (unselect) {

0 commit comments

Comments
 (0)