Skip to content

Commit

Permalink
fix(runtime-core/template-ref): template ref used in the same templat…
Browse files Browse the repository at this point in the history
…e should trigger update

fix #1505
  • Loading branch information
yyx990803 committed Jul 6, 2020
1 parent 64e2f46 commit 36b6b4f
Show file tree
Hide file tree
Showing 3 changed files with 38 additions and 8 deletions.
24 changes: 23 additions & 1 deletion packages/runtime-core/__tests__/apiTemplateRef.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,8 @@ import {
render,
nextTick,
defineComponent,
reactive
reactive,
serializeInner
} from '@vue/runtime-test'

// reference: https://vue-composition-api-rfc.netlify.com/api.html#template-refs
Expand Down Expand Up @@ -246,4 +247,25 @@ describe('api: template refs', () => {
expect(refKey2.value).toBe(root.children[2])
expect(refKey3.value).toBe(root.children[3])
})

// #1505
test('reactive template ref in the same template', async () => {
const Comp = {
setup() {
const el = ref()
return { el }
},
render(this: any) {
return h('div', { id: 'foo', ref: 'el' }, this.el && this.el.props.id)
}
}

const root = nodeOps.createElement('div')
render(h(Comp), root)
// ref not ready on first render, but should queue an update immediately
expect(serializeInner(root)).toBe(`<div id="foo"></div>`)
await nextTick()
// ref should be updated
expect(serializeInner(root)).toBe(`<div id="foo">foo</div>`)
})
})
2 changes: 1 addition & 1 deletion packages/runtime-core/src/hydration.ts
Original file line number Diff line number Diff line change
Expand Up @@ -227,7 +227,7 @@ export function createHydrationFunctions(
}

if (ref != null && parentComponent) {
setRef(ref, null, parentComponent, vnode)
setRef(ref, null, parentComponent, parentSuspense, vnode)
}

return nextNode
Expand Down
20 changes: 14 additions & 6 deletions packages/runtime-core/src/renderer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -275,7 +275,8 @@ export const queuePostRenderEffect = __FEATURE_SUSPENSE__
export const setRef = (
rawRef: VNodeNormalizedRef,
oldRawRef: VNodeNormalizedRef | null,
parent: ComponentInternalInstance,
parentComponent: ComponentInternalInstance,
parentSuspense: SuspenseBoundary | null,
vnode: VNode | null
) => {
let value: ComponentPublicInstance | RendererNode | null
Expand Down Expand Up @@ -306,7 +307,9 @@ export const setRef = (
if (isString(oldRef)) {
refs[oldRef] = null
if (hasOwn(setupState, oldRef)) {
setupState[oldRef] = null
queuePostRenderEffect(() => {
setupState[oldRef] = null
}, parentSuspense)
}
} else if (isRef(oldRef)) {
oldRef.value = null
Expand All @@ -316,12 +319,17 @@ export const setRef = (
if (isString(ref)) {
refs[ref] = value
if (hasOwn(setupState, ref)) {
setupState[ref] = value
queuePostRenderEffect(() => {
setupState[ref] = value
}, parentSuspense)
}
} else if (isRef(ref)) {
ref.value = value
} else if (isFunction(ref)) {
callWithErrorHandling(ref, parent, ErrorCodes.FUNCTION_REF, [value, refs])
callWithErrorHandling(ref, parentComponent, ErrorCodes.FUNCTION_REF, [
value,
refs
])
} else if (__DEV__) {
warn('Invalid template ref type:', value, `(${typeof value})`)
}
Expand Down Expand Up @@ -497,7 +505,7 @@ function baseCreateRenderer(

// set ref
if (ref != null && parentComponent) {
setRef(ref, n1 && n1.ref, parentComponent, n2)
setRef(ref, n1 && n1.ref, parentComponent, parentSuspense, n2)
}
}

Expand Down Expand Up @@ -1868,7 +1876,7 @@ function baseCreateRenderer(
} = vnode
// unset ref
if (ref != null && parentComponent) {
setRef(ref, null, parentComponent, null)
setRef(ref, null, parentComponent, parentSuspense, null)
}

if (shapeFlag & ShapeFlags.COMPONENT_SHOULD_KEEP_ALIVE) {
Expand Down

0 comments on commit 36b6b4f

Please sign in to comment.