Skip to content

Commit

Permalink
fix(runtime-core): should disable tracking inside directive lifecycle…
Browse files Browse the repository at this point in the history
… hooks (#3699)
  • Loading branch information
HcySunYang authored May 27, 2021
1 parent 9f24195 commit ff50e8d
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 0 deletions.
28 changes: 28 additions & 0 deletions packages/runtime-core/__tests__/directives.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -367,4 +367,32 @@ describe('directives', () => {
expect(d1.mounted).toHaveBeenCalled()
expect(d2.mounted).toHaveBeenCalled()
})

test('should disable tracking inside directive lifecycle hooks', async () => {
const count = ref(0)
const text = ref('')
const beforeUpdate = jest.fn(() => count.value++)

const App = {
render() {
return withDirectives(h('p', text.value), [
[
{
beforeUpdate
}
]
])
}
}

const root = nodeOps.createElement('div')
render(h(App), root)
expect(beforeUpdate).toHaveBeenCalledTimes(0)
expect(count.value).toBe(0)

text.value = 'foo'
await nextTick()
expect(beforeUpdate).toHaveBeenCalledTimes(1)
expect(count.value).toBe(1)
})
})
5 changes: 5 additions & 0 deletions packages/runtime-core/src/directives.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import { currentRenderingInstance } from './componentRenderContext'
import { callWithAsyncErrorHandling, ErrorCodes } from './errorHandling'
import { ComponentPublicInstance } from './componentPublicInstance'
import { mapCompatDirectiveHook } from './compat/customDirective'
import { pauseTracking, resetTracking } from '@vue/reactivity'

export interface DirectiveBinding<V = any> {
instance: ComponentPublicInstance | null
Expand Down Expand Up @@ -130,12 +131,16 @@ export function invokeDirectiveHook(
hook = mapCompatDirectiveHook(name, binding.dir, instance)
}
if (hook) {
// disable tracking inside all lifecycle hooks
// since they can potentially be called inside effects.
pauseTracking()
callWithAsyncErrorHandling(hook, instance, ErrorCodes.DIRECTIVE_HOOK, [
vnode.el,
binding,
vnode,
prevVNode
])
resetTracking()
}
}
}

0 comments on commit ff50e8d

Please sign in to comment.