Skip to content

Commit 42239cf

Browse files
committedNov 10, 2022
fix(teleport/css-v-bind): fix css v-bind for teleported content
fix #4605 close #4609 (used tests from this PR)
1 parent 9ca8b7c commit 42239cf

File tree

4 files changed

+94
-2
lines changed

4 files changed

+94
-2
lines changed
 

‎packages/runtime-core/src/component.ts

+6
Original file line numberDiff line numberDiff line change
@@ -458,6 +458,12 @@ export interface ComponentInternalInstance {
458458
* @internal
459459
*/
460460
n?: () => Promise<void>
461+
/**
462+
* `updateTeleportCssVars`
463+
* For updating css vars on contained teleports
464+
* @internal
465+
*/
466+
ut?: (vars?: Record<string, string>) => void
461467
}
462468

463469
const emptyAppContext = createAppContext()

‎packages/runtime-core/src/components/Teleport.ts

+20
Original file line numberDiff line numberDiff line change
@@ -222,6 +222,8 @@ export const TeleportImpl = {
222222
}
223223
}
224224
}
225+
226+
updateCssVars(parentComponent, n2)
225227
},
226228

227229
remove(
@@ -383,6 +385,7 @@ function hydrateTeleport(
383385
)
384386
}
385387
}
388+
updateCssVars(parentComponent, vnode)
386389
}
387390
return vnode.anchor && nextSibling(vnode.anchor as Node)
388391
}
@@ -392,3 +395,20 @@ export const Teleport = TeleportImpl as unknown as {
392395
__isTeleport: true
393396
new (): { $props: VNodeProps & TeleportProps }
394397
}
398+
399+
function updateCssVars(
400+
parentComponent: ComponentInternalInstance | null,
401+
vnode: VNode
402+
) {
403+
// presence of .ut method indicates owner component uses css vars.
404+
// code path here can assume browser environment.
405+
if (parentComponent && parentComponent.ut) {
406+
let node = (vnode.children as VNode[])[0].el!
407+
while (node !== vnode.targetAnchor) {
408+
if (node.nodeType === 1)
409+
node.setAttribute('data-v-owner', parentComponent.uid)
410+
node = node.nextSibling
411+
}
412+
parentComponent.ut()
413+
}
414+
}

‎packages/runtime-dom/__tests__/helpers/useCssVars.spec.ts

+55
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import {
88
nextTick,
99
ComponentOptions,
1010
Suspense,
11+
Teleport,
1112
FunctionalComponent
1213
} from '@vue/runtime-dom'
1314

@@ -196,4 +197,58 @@ describe('useCssVars', () => {
196197
expect((c as HTMLElement).style.getPropertyValue(`--color`)).toBe('red')
197198
}
198199
})
200+
201+
test('with teleport', async () => {
202+
document.body.innerHTML = ''
203+
const state = reactive({ color: 'red' })
204+
const root = document.createElement('div')
205+
const target = document.body
206+
207+
const App = {
208+
setup() {
209+
useCssVars(() => state)
210+
return () => [h(Teleport, { to: target }, [h('div')])]
211+
}
212+
}
213+
214+
render(h(App), root)
215+
await nextTick()
216+
for (const c of [].slice.call(target.children as any)) {
217+
expect((c as HTMLElement).style.getPropertyValue(`--color`)).toBe('red')
218+
}
219+
})
220+
221+
test('with teleport(change subTree)', async () => {
222+
document.body.innerHTML = ''
223+
const state = reactive({ color: 'red' })
224+
const root = document.createElement('div')
225+
const target = document.body
226+
const toggle = ref(false)
227+
228+
const App = {
229+
setup() {
230+
useCssVars(() => state)
231+
return () => [
232+
h(Teleport, { to: target }, [
233+
h('div'),
234+
toggle.value ? h('div') : null
235+
])
236+
]
237+
}
238+
}
239+
240+
render(h(App), root)
241+
await nextTick()
242+
expect(target.children.length).toBe(1)
243+
for (const c of [].slice.call(target.children as any)) {
244+
expect((c as HTMLElement).style.getPropertyValue(`--color`)).toBe('red')
245+
}
246+
247+
toggle.value = true
248+
await nextTick()
249+
expect(target.children.length).toBe(2)
250+
for (const c of [].slice.call(target.children as any)) {
251+
expect((c as HTMLElement).style.getPropertyValue(`--color`)).toBe('red')
252+
}
253+
})
199254
})

‎packages/runtime-dom/src/helpers/useCssVars.ts

+13-2
Original file line numberDiff line numberDiff line change
@@ -25,9 +25,20 @@ export function useCssVars(getter: (ctx: any) => Record<string, string>) {
2525
return
2626
}
2727

28-
const setVars = () =>
29-
setVarsOnVNode(instance.subTree, getter(instance.proxy!))
28+
const updateTeleports = (instance.ut = (vars = getter(instance.proxy)) => {
29+
Array.from(
30+
document.querySelectorAll(`[data-v-owner="${instance.uid}"]`)
31+
).forEach(node => setVarsOnNode(node, vars))
32+
})
33+
34+
const setVars = () => {
35+
const vars = getter(instance.proxy)
36+
setVarsOnVNode(instance.subTree, vars)
37+
updateTeleports(vars)
38+
}
39+
3040
watchPostEffect(setVars)
41+
3142
onMounted(() => {
3243
const ob = new MutationObserver(setVars)
3344
ob.observe(instance.subTree.el!.parentNode, { childList: true })

0 commit comments

Comments
 (0)
Please sign in to comment.