2020-04-03 23:08:17 +00:00
|
|
|
import {
|
2020-10-19 21:15:53 +00:00
|
|
|
camelize,
|
2020-04-03 23:08:17 +00:00
|
|
|
EMPTY_OBJ,
|
2020-10-19 21:15:53 +00:00
|
|
|
toHandlerKey,
|
|
|
|
extend,
|
|
|
|
hasOwn,
|
2020-04-04 00:40:34 +00:00
|
|
|
hyphenate,
|
2020-10-19 21:15:53 +00:00
|
|
|
isArray,
|
2020-04-06 21:57:27 +00:00
|
|
|
isFunction,
|
2020-10-20 13:59:27 +00:00
|
|
|
isOn,
|
|
|
|
toNumber
|
2020-04-03 23:08:17 +00:00
|
|
|
} from '@vue/shared'
|
2020-08-31 22:32:07 +00:00
|
|
|
import {
|
|
|
|
ComponentInternalInstance,
|
|
|
|
ComponentOptions,
|
2020-10-06 22:28:56 +00:00
|
|
|
ConcreteComponent,
|
|
|
|
formatComponentName
|
2020-08-31 22:32:07 +00:00
|
|
|
} from './component'
|
2020-04-03 23:08:17 +00:00
|
|
|
import { callWithAsyncErrorHandling, ErrorCodes } from './errorHandling'
|
2020-04-04 00:40:34 +00:00
|
|
|
import { warn } from './warning'
|
2020-06-09 14:37:00 +00:00
|
|
|
import { UnionToIntersection } from './helpers/typeUtils'
|
2020-08-23 23:31:32 +00:00
|
|
|
import { devtoolsComponentEmit } from './devtools'
|
2020-08-31 22:32:07 +00:00
|
|
|
import { AppContext } from './apiCreateApp'
|
2020-04-03 23:08:17 +00:00
|
|
|
|
|
|
|
export type ObjectEmitsOptions = Record<
|
|
|
|
string,
|
|
|
|
((...args: any[]) => any) | null
|
|
|
|
>
|
|
|
|
export type EmitsOptions = ObjectEmitsOptions | string[]
|
|
|
|
|
|
|
|
export type EmitFn<
|
|
|
|
Options = ObjectEmitsOptions,
|
|
|
|
Event extends keyof Options = keyof Options
|
2020-09-22 14:05:37 +00:00
|
|
|
> = Options extends Array<infer V>
|
|
|
|
? (event: V, ...args: any[]) => void
|
2020-07-08 15:51:03 +00:00
|
|
|
: {} extends Options // if the emit is empty object (usually the default value for emit) should be converted to function
|
|
|
|
? (event: string, ...args: any[]) => void
|
|
|
|
: UnionToIntersection<
|
|
|
|
{
|
|
|
|
[key in Event]: Options[key] extends ((...args: infer Args) => any)
|
|
|
|
? (event: key, ...args: Args) => void
|
|
|
|
: (event: key, ...args: any[]) => void
|
|
|
|
}[Event]
|
|
|
|
>
|
2020-04-03 23:08:17 +00:00
|
|
|
|
|
|
|
export function emit(
|
|
|
|
instance: ComponentInternalInstance,
|
|
|
|
event: string,
|
2020-10-20 13:59:27 +00:00
|
|
|
...rawArgs: any[]
|
2020-04-10 14:59:46 +00:00
|
|
|
) {
|
2020-04-03 23:08:17 +00:00
|
|
|
const props = instance.vnode.props || EMPTY_OBJ
|
2020-04-04 00:40:34 +00:00
|
|
|
|
|
|
|
if (__DEV__) {
|
2020-08-31 22:32:07 +00:00
|
|
|
const {
|
|
|
|
emitsOptions,
|
|
|
|
propsOptions: [propsOptions]
|
|
|
|
} = instance
|
|
|
|
if (emitsOptions) {
|
|
|
|
if (!(event in emitsOptions)) {
|
2020-10-19 21:15:53 +00:00
|
|
|
if (!propsOptions || !(toHandlerKey(event) in propsOptions)) {
|
2020-04-16 15:27:52 +00:00
|
|
|
warn(
|
|
|
|
`Component emitted event "${event}" but it is neither declared in ` +
|
2020-10-19 21:15:53 +00:00
|
|
|
`the emits option nor as an "${toHandlerKey(event)}" prop.`
|
2020-04-16 15:27:52 +00:00
|
|
|
)
|
|
|
|
}
|
2020-04-04 00:40:34 +00:00
|
|
|
} else {
|
2020-08-31 22:32:07 +00:00
|
|
|
const validator = emitsOptions[event]
|
2020-04-04 00:40:34 +00:00
|
|
|
if (isFunction(validator)) {
|
2020-10-20 13:59:27 +00:00
|
|
|
const isValid = validator(...rawArgs)
|
2020-04-04 00:40:34 +00:00
|
|
|
if (!isValid) {
|
|
|
|
warn(
|
|
|
|
`Invalid event arguments: event validation failed for event "${event}".`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-20 13:59:27 +00:00
|
|
|
let args = rawArgs
|
|
|
|
const isModelListener = event.startsWith('update:')
|
|
|
|
|
|
|
|
// for v-model update:xxx events, apply modifiers on args
|
|
|
|
const modelArg = isModelListener && event.slice(7)
|
|
|
|
if (modelArg && modelArg in props) {
|
|
|
|
const modifiersKey = `${
|
|
|
|
modelArg === 'modelValue' ? 'model' : modelArg
|
|
|
|
}Modifiers`
|
|
|
|
const { number, trim } = props[modifiersKey] || EMPTY_OBJ
|
|
|
|
if (trim) {
|
|
|
|
args = rawArgs.map(a => a.trim())
|
|
|
|
} else if (number) {
|
|
|
|
args = rawArgs.map(toNumber)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-23 23:31:32 +00:00
|
|
|
if (__DEV__ || __FEATURE_PROD_DEVTOOLS__) {
|
|
|
|
devtoolsComponentEmit(instance, event, args)
|
|
|
|
}
|
|
|
|
|
2020-10-06 22:28:56 +00:00
|
|
|
if (__DEV__) {
|
|
|
|
const lowerCaseEvent = event.toLowerCase()
|
2020-10-19 21:15:53 +00:00
|
|
|
if (lowerCaseEvent !== event && props[toHandlerKey(lowerCaseEvent)]) {
|
2020-10-06 22:28:56 +00:00
|
|
|
warn(
|
|
|
|
`Event "${lowerCaseEvent}" is emitted in component ` +
|
|
|
|
`${formatComponentName(
|
|
|
|
instance,
|
|
|
|
instance.type
|
|
|
|
)} but the handler is registered for "${event}". ` +
|
|
|
|
`Note that HTML attributes are case-insensitive and you cannot use ` +
|
|
|
|
`v-on to listen to camelCase events when using in-DOM templates. ` +
|
|
|
|
`You should probably use "${hyphenate(event)}" instead of "${event}".`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// convert handler name to camelCase. See issue #2249
|
2020-10-19 21:15:53 +00:00
|
|
|
let handlerName = toHandlerKey(camelize(event))
|
2020-07-14 15:48:05 +00:00
|
|
|
let handler = props[handlerName]
|
2020-04-03 23:08:17 +00:00
|
|
|
// for v-model update:xxx events, also trigger kebab-case equivalent
|
|
|
|
// for props passed via kebab-case
|
2020-10-20 13:59:27 +00:00
|
|
|
if (!handler && isModelListener) {
|
2020-10-19 21:15:53 +00:00
|
|
|
handlerName = toHandlerKey(hyphenate(event))
|
2020-07-14 15:48:05 +00:00
|
|
|
handler = props[handlerName]
|
|
|
|
}
|
2020-10-20 13:49:53 +00:00
|
|
|
|
|
|
|
if (handler) {
|
|
|
|
callWithAsyncErrorHandling(
|
|
|
|
handler,
|
|
|
|
instance,
|
|
|
|
ErrorCodes.COMPONENT_EVENT_HANDLER,
|
|
|
|
args
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
const onceHandler = props[handlerName + `Once`]
|
|
|
|
if (onceHandler) {
|
2020-07-14 15:48:05 +00:00
|
|
|
if (!instance.emitted) {
|
|
|
|
;(instance.emitted = {} as Record<string, boolean>)[handlerName] = true
|
|
|
|
} else if (instance.emitted[handlerName]) {
|
|
|
|
return
|
|
|
|
}
|
2020-04-10 14:59:46 +00:00
|
|
|
callWithAsyncErrorHandling(
|
2020-10-20 13:49:53 +00:00
|
|
|
onceHandler,
|
2020-04-03 23:08:17 +00:00
|
|
|
instance,
|
|
|
|
ErrorCodes.COMPONENT_EVENT_HANDLER,
|
|
|
|
args
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-31 22:32:07 +00:00
|
|
|
export function normalizeEmitsOptions(
|
|
|
|
comp: ConcreteComponent,
|
|
|
|
appContext: AppContext,
|
|
|
|
asMixin = false
|
|
|
|
): ObjectEmitsOptions | null {
|
2020-10-06 19:31:29 +00:00
|
|
|
if (!appContext.deopt && comp.__emits !== undefined) {
|
|
|
|
return comp.__emits
|
2020-07-13 15:55:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const raw = comp.emits
|
|
|
|
let normalized: ObjectEmitsOptions = {}
|
|
|
|
|
|
|
|
// apply mixin/extends props
|
|
|
|
let hasExtends = false
|
2020-07-21 01:51:30 +00:00
|
|
|
if (__FEATURE_OPTIONS_API__ && !isFunction(comp)) {
|
2020-08-31 22:32:07 +00:00
|
|
|
const extendEmits = (raw: ComponentOptions) => {
|
2020-07-13 15:55:46 +00:00
|
|
|
hasExtends = true
|
2020-08-31 22:32:07 +00:00
|
|
|
extend(normalized, normalizeEmitsOptions(raw, appContext, true))
|
|
|
|
}
|
|
|
|
if (!asMixin && appContext.mixins.length) {
|
|
|
|
appContext.mixins.forEach(extendEmits)
|
|
|
|
}
|
|
|
|
if (comp.extends) {
|
|
|
|
extendEmits(comp.extends)
|
2020-04-03 23:08:17 +00:00
|
|
|
}
|
2020-07-13 15:55:46 +00:00
|
|
|
if (comp.mixins) {
|
2020-08-31 22:32:07 +00:00
|
|
|
comp.mixins.forEach(extendEmits)
|
2020-07-13 15:55:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!raw && !hasExtends) {
|
2020-10-06 19:31:29 +00:00
|
|
|
return (comp.__emits = null)
|
2020-07-13 15:55:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (isArray(raw)) {
|
|
|
|
raw.forEach(key => (normalized[key] = null))
|
2020-04-03 23:08:17 +00:00
|
|
|
} else {
|
2020-07-13 15:55:46 +00:00
|
|
|
extend(normalized, raw)
|
2020-04-03 23:08:17 +00:00
|
|
|
}
|
2020-10-06 19:31:29 +00:00
|
|
|
return (comp.__emits = normalized)
|
2020-04-03 23:08:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check if an incoming prop key is a declared emit event listener.
|
|
|
|
// e.g. With `emits: { click: null }`, props named `onClick` and `onclick` are
|
|
|
|
// both considered matched listeners.
|
2020-08-31 22:32:07 +00:00
|
|
|
export function isEmitListener(
|
|
|
|
options: ObjectEmitsOptions | null,
|
|
|
|
key: string
|
|
|
|
): boolean {
|
|
|
|
if (!options || !isOn(key)) {
|
2020-07-13 15:55:46 +00:00
|
|
|
return false
|
|
|
|
}
|
2020-07-14 17:20:59 +00:00
|
|
|
key = key.replace(/Once$/, '')
|
2020-04-03 23:08:17 +00:00
|
|
|
return (
|
2020-08-31 22:32:07 +00:00
|
|
|
hasOwn(options, key[2].toLowerCase() + key.slice(3)) ||
|
|
|
|
hasOwn(options, key.slice(2))
|
2020-04-03 23:08:17 +00:00
|
|
|
)
|
|
|
|
}
|