fix(runtime-core/emits): merge emits options from mixins/extends
fix #1562
This commit is contained in:
parent
c2d3da9dc4
commit
ba3b3cdda9
@ -143,12 +143,47 @@ describe('component: emit', () => {
|
|||||||
expect(`event validation failed for event "foo"`).toHaveBeenWarned()
|
expect(`event validation failed for event "foo"`).toHaveBeenWarned()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('merging from mixins', () => {
|
||||||
|
const mixin = {
|
||||||
|
emits: {
|
||||||
|
foo: (arg: number) => arg > 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const Foo = defineComponent({
|
||||||
|
mixins: [mixin],
|
||||||
|
render() {},
|
||||||
|
created() {
|
||||||
|
this.$emit('foo', -1)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
render(h(Foo), nodeOps.createElement('div'))
|
||||||
|
expect(`event validation failed for event "foo"`).toHaveBeenWarned()
|
||||||
|
})
|
||||||
|
|
||||||
test('isEmitListener', () => {
|
test('isEmitListener', () => {
|
||||||
expect(isEmitListener(['click'], 'onClick')).toBe(true)
|
const def1 = { emits: ['click'] }
|
||||||
expect(isEmitListener(['click'], 'onclick')).toBe(false)
|
expect(isEmitListener(def1, 'onClick')).toBe(true)
|
||||||
expect(isEmitListener({ click: null }, 'onClick')).toBe(true)
|
expect(isEmitListener(def1, 'onclick')).toBe(false)
|
||||||
expect(isEmitListener({ click: null }, 'onclick')).toBe(false)
|
expect(isEmitListener(def1, 'onBlick')).toBe(false)
|
||||||
expect(isEmitListener(['click'], 'onBlick')).toBe(false)
|
|
||||||
expect(isEmitListener({ click: null }, 'onBlick')).toBe(false)
|
const def2 = { emits: { click: null } }
|
||||||
|
expect(isEmitListener(def2, 'onClick')).toBe(true)
|
||||||
|
expect(isEmitListener(def2, 'onclick')).toBe(false)
|
||||||
|
expect(isEmitListener(def2, 'onBlick')).toBe(false)
|
||||||
|
|
||||||
|
const mixin1 = { emits: ['foo'] }
|
||||||
|
const mixin2 = { emits: ['bar'] }
|
||||||
|
const extend = { emits: ['baz'] }
|
||||||
|
const def3 = {
|
||||||
|
emits: { click: null },
|
||||||
|
mixins: [mixin1, mixin2],
|
||||||
|
extends: extend
|
||||||
|
}
|
||||||
|
expect(isEmitListener(def3, 'onClick')).toBe(true)
|
||||||
|
expect(isEmitListener(def3, 'onFoo')).toBe(true)
|
||||||
|
expect(isEmitListener(def3, 'onBar')).toBe(true)
|
||||||
|
expect(isEmitListener(def3, 'onBaz')).toBe(true)
|
||||||
|
expect(isEmitListener(def3, 'onclick')).toBe(false)
|
||||||
|
expect(isEmitListener(def3, 'onBlick')).toBe(false)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -59,6 +59,10 @@ export interface ComponentInternalOptions {
|
|||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
__props?: NormalizedPropsOptions | []
|
__props?: NormalizedPropsOptions | []
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
*/
|
||||||
|
__emits?: ObjectEmitsOptions
|
||||||
/**
|
/**
|
||||||
* @internal
|
* @internal
|
||||||
*/
|
*/
|
||||||
|
@ -6,9 +6,9 @@ import {
|
|||||||
capitalize,
|
capitalize,
|
||||||
hyphenate,
|
hyphenate,
|
||||||
isFunction,
|
isFunction,
|
||||||
def
|
extend
|
||||||
} from '@vue/shared'
|
} from '@vue/shared'
|
||||||
import { ComponentInternalInstance } from './component'
|
import { ComponentInternalInstance, Component } from './component'
|
||||||
import { callWithAsyncErrorHandling, ErrorCodes } from './errorHandling'
|
import { callWithAsyncErrorHandling, ErrorCodes } from './errorHandling'
|
||||||
import { warn } from './warning'
|
import { warn } from './warning'
|
||||||
import { normalizePropsOptions } from './componentProps'
|
import { normalizePropsOptions } from './componentProps'
|
||||||
@ -43,7 +43,7 @@ export function emit(
|
|||||||
const props = instance.vnode.props || EMPTY_OBJ
|
const props = instance.vnode.props || EMPTY_OBJ
|
||||||
|
|
||||||
if (__DEV__) {
|
if (__DEV__) {
|
||||||
const options = normalizeEmitsOptions(instance.type.emits)
|
const options = normalizeEmitsOptions(instance.type)
|
||||||
if (options) {
|
if (options) {
|
||||||
if (!(event in options)) {
|
if (!(event in options)) {
|
||||||
const propsOptions = normalizePropsOptions(instance.type)[0]
|
const propsOptions = normalizePropsOptions(instance.type)[0]
|
||||||
@ -84,34 +84,52 @@ export function emit(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function normalizeEmitsOptions(
|
function normalizeEmitsOptions(
|
||||||
options: EmitsOptions | undefined
|
comp: Component
|
||||||
): ObjectEmitsOptions | undefined {
|
): ObjectEmitsOptions | undefined {
|
||||||
if (!options) {
|
if (hasOwn(comp, '__emits')) {
|
||||||
return
|
return comp.__emits
|
||||||
} else if (isArray(options)) {
|
|
||||||
if ((options as any)._n) {
|
|
||||||
return (options as any)._n
|
|
||||||
}
|
|
||||||
const normalized: ObjectEmitsOptions = {}
|
|
||||||
options.forEach(key => (normalized[key] = null))
|
|
||||||
def(options, '_n', normalized)
|
|
||||||
return normalized
|
|
||||||
} else {
|
|
||||||
return options
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const raw = comp.emits
|
||||||
|
let normalized: ObjectEmitsOptions = {}
|
||||||
|
|
||||||
|
// apply mixin/extends props
|
||||||
|
let hasExtends = false
|
||||||
|
if (__FEATURE_OPTIONS__ && !isFunction(comp)) {
|
||||||
|
if (comp.extends) {
|
||||||
|
hasExtends = true
|
||||||
|
extend(normalized, normalizeEmitsOptions(comp.extends))
|
||||||
|
}
|
||||||
|
if (comp.mixins) {
|
||||||
|
hasExtends = true
|
||||||
|
comp.mixins.forEach(m => extend(normalized, normalizeEmitsOptions(m)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!raw && !hasExtends) {
|
||||||
|
return (comp.__emits = undefined)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isArray(raw)) {
|
||||||
|
raw.forEach(key => (normalized[key] = null))
|
||||||
|
} else {
|
||||||
|
extend(normalized, raw)
|
||||||
|
}
|
||||||
|
return (comp.__emits = normalized)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if an incoming prop key is a declared emit event listener.
|
// Check if an incoming prop key is a declared emit event listener.
|
||||||
// e.g. With `emits: { click: null }`, props named `onClick` and `onclick` are
|
// e.g. With `emits: { click: null }`, props named `onClick` and `onclick` are
|
||||||
// both considered matched listeners.
|
// both considered matched listeners.
|
||||||
export function isEmitListener(emits: EmitsOptions, key: string): boolean {
|
export function isEmitListener(comp: Component, key: string): boolean {
|
||||||
|
if (!isOn(key)) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
const emits = normalizeEmitsOptions(comp)
|
||||||
return (
|
return (
|
||||||
isOn(key) &&
|
!!emits &&
|
||||||
(hasOwn(
|
(hasOwn(emits, key[2].toLowerCase() + key.slice(3)) ||
|
||||||
(emits = normalizeEmitsOptions(emits) as ObjectEmitsOptions),
|
|
||||||
key[2].toLowerCase() + key.slice(3)
|
|
||||||
) ||
|
|
||||||
hasOwn(emits, key.slice(2)))
|
hasOwn(emits, key.slice(2)))
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -242,8 +242,6 @@ function setFullProps(
|
|||||||
attrs: Data
|
attrs: Data
|
||||||
) {
|
) {
|
||||||
const [options, needCastKeys] = normalizePropsOptions(instance.type)
|
const [options, needCastKeys] = normalizePropsOptions(instance.type)
|
||||||
const emits = instance.type.emits
|
|
||||||
|
|
||||||
if (rawProps) {
|
if (rawProps) {
|
||||||
for (const key in rawProps) {
|
for (const key in rawProps) {
|
||||||
const value = rawProps[key]
|
const value = rawProps[key]
|
||||||
@ -256,7 +254,7 @@ function setFullProps(
|
|||||||
let camelKey
|
let camelKey
|
||||||
if (options && hasOwn(options, (camelKey = camelize(key)))) {
|
if (options && hasOwn(options, (camelKey = camelize(key)))) {
|
||||||
props[camelKey] = value
|
props[camelKey] = value
|
||||||
} else if (!emits || !isEmitListener(emits, key)) {
|
} else if (!isEmitListener(instance.type, key)) {
|
||||||
// Any non-declared (either as a prop or an emitted event) props are put
|
// Any non-declared (either as a prop or an emitted event) props are put
|
||||||
// into a separate `attrs` object for spreading. Make sure to preserve
|
// into a separate `attrs` object for spreading. Make sure to preserve
|
||||||
// original key casing
|
// original key casing
|
||||||
|
Loading…
x
Reference in New Issue
Block a user