refactor(directives): remove binding.instance

BREAKING CHANGE: custom directive bindings no longer expose instance

    This is a rarely used property that creates extra complexity in
    ensuring it points to the correct instance. From a design
    perspective, a custom directive should be scoped to the element and
    data it is bound to and should not have access to the entire
    instance in the first place.
This commit is contained in:
Evan You 2020-02-25 19:35:48 -05:00
parent 3eab143843
commit 52cc7e8231
2 changed files with 2 additions and 27 deletions

View File

@ -9,7 +9,6 @@ import {
DirectiveBinding, DirectiveBinding,
nextTick nextTick
} from '@vue/runtime-test' } from '@vue/runtime-test'
import { currentInstance, ComponentInternalInstance } from '../src/component'
describe('directives', () => { describe('directives', () => {
it('should work', async () => { it('should work', async () => {
@ -18,7 +17,6 @@ describe('directives', () => {
function assertBindings(binding: DirectiveBinding) { function assertBindings(binding: DirectiveBinding) {
expect(binding.value).toBe(count.value) expect(binding.value).toBe(count.value)
expect(binding.arg).toBe('foo') expect(binding.arg).toBe('foo')
expect(binding.instance).toBe(_instance && _instance.proxy)
expect(binding.modifiers && binding.modifiers.ok).toBe(true) expect(binding.modifiers && binding.modifiers.ok).toBe(true)
} }
@ -107,13 +105,9 @@ describe('directives', () => {
unmounted unmounted
} }
let _instance: ComponentInternalInstance | null = null
let _vnode: VNode | null = null let _vnode: VNode | null = null
let _prevVnode: VNode | null = null let _prevVnode: VNode | null = null
const Comp = { const Comp = {
setup() {
_instance = currentInstance
},
render() { render() {
_prevVnode = _vnode _prevVnode = _vnode
_vnode = withDirectives(h('div', count.value), [ _vnode = withDirectives(h('div', count.value), [
@ -153,7 +147,6 @@ describe('directives', () => {
function assertBindings(binding: DirectiveBinding) { function assertBindings(binding: DirectiveBinding) {
expect(binding.value).toBe(count.value) expect(binding.value).toBe(count.value)
expect(binding.arg).toBe('foo') expect(binding.arg).toBe('foo')
expect(binding.instance).toBe(_instance && _instance.proxy)
expect(binding.modifiers && binding.modifiers.ok).toBe(true) expect(binding.modifiers && binding.modifiers.ok).toBe(true)
} }
@ -167,13 +160,9 @@ describe('directives', () => {
expect(prevVNode).toBe(_prevVnode) expect(prevVNode).toBe(_prevVnode)
}) as DirectiveHook) }) as DirectiveHook)
let _instance: ComponentInternalInstance | null = null
let _vnode: VNode | null = null let _vnode: VNode | null = null
let _prevVnode: VNode | null = null let _prevVnode: VNode | null = null
const Comp = { const Comp = {
setup() {
_instance = currentInstance
},
render() { render() {
_prevVnode = _vnode _prevVnode = _vnode
_vnode = withDirectives(h('div', count.value), [ _vnode = withDirectives(h('div', count.value), [
@ -207,7 +196,6 @@ describe('directives', () => {
function assertBindings(binding: DirectiveBinding) { function assertBindings(binding: DirectiveBinding) {
expect(binding.value).toBe(count.value) expect(binding.value).toBe(count.value)
expect(binding.arg).toBe('foo') expect(binding.arg).toBe('foo')
expect(binding.instance).toBe(_instance && _instance.proxy)
expect(binding.modifiers && binding.modifiers.ok).toBe(true) expect(binding.modifiers && binding.modifiers.ok).toBe(true)
} }
@ -296,7 +284,6 @@ describe('directives', () => {
unmounted unmounted
} }
let _instance: ComponentInternalInstance | null = null
let _vnode: VNode | null = null let _vnode: VNode | null = null
let _prevVnode: VNode | null = null let _prevVnode: VNode | null = null
@ -307,9 +294,6 @@ describe('directives', () => {
} }
const Comp = { const Comp = {
setup() {
_instance = currentInstance
},
render() { render() {
return withDirectives(h(Child, { count: count.value }), [ return withDirectives(h(Child, { count: count.value }), [
[ [

View File

@ -15,12 +15,9 @@ import { VNode } from './vnode'
import { isFunction, EMPTY_OBJ, makeMap, EMPTY_ARR } from '@vue/shared' import { isFunction, EMPTY_OBJ, makeMap, EMPTY_ARR } from '@vue/shared'
import { warn } from './warning' import { warn } from './warning'
import { ComponentInternalInstance } from './component' import { ComponentInternalInstance } from './component'
import { currentRenderingInstance } from './componentRenderUtils'
import { callWithAsyncErrorHandling, ErrorCodes } from './errorHandling' import { callWithAsyncErrorHandling, ErrorCodes } from './errorHandling'
import { ComponentPublicInstance } from './componentProxy'
export interface DirectiveBinding { export interface DirectiveBinding {
instance: ComponentPublicInstance | null
value: any value: any
oldValue: any oldValue: any
arg?: string arg?: string
@ -108,14 +105,9 @@ export function withDirectives<T extends VNode>(
vnode: T, vnode: T,
directives: DirectiveArguments directives: DirectiveArguments
): T { ): T {
const internalInstance = currentRenderingInstance
if (internalInstance === null) {
__DEV__ && warn(`withDirectives can only be used inside render functions.`)
return vnode
}
const instance = internalInstance.proxy
const props = vnode.props || (vnode.props = {}) const props = vnode.props || (vnode.props = {})
const bindings = vnode.dirs || (vnode.dirs = new Array(directives.length)) const bindings: DirectiveBinding[] =
vnode.dirs || (vnode.dirs = new Array(directives.length))
const injected: Record<string, true> = {} const injected: Record<string, true> = {}
for (let i = 0; i < directives.length; i++) { for (let i = 0; i < directives.length; i++) {
let [dir, value, arg, modifiers = EMPTY_OBJ] = directives[i] let [dir, value, arg, modifiers = EMPTY_OBJ] = directives[i]
@ -127,7 +119,6 @@ export function withDirectives<T extends VNode>(
} }
bindings[i] = { bindings[i] = {
dir, dir,
instance,
value, value,
oldValue: void 0, oldValue: void 0,
arg, arg,