2019-08-30 14:36:30 +00:00
|
|
|
import { VNode } from './vnode'
|
2020-04-02 01:36:50 +00:00
|
|
|
import {
|
|
|
|
Data,
|
|
|
|
ComponentInternalInstance,
|
2020-08-19 20:11:29 +00:00
|
|
|
ConcreteComponent,
|
2020-04-02 01:36:50 +00:00
|
|
|
formatComponentName
|
|
|
|
} from './component'
|
2019-10-08 13:26:09 +00:00
|
|
|
import { isString, isFunction } from '@vue/shared'
|
2020-02-18 04:14:07 +00:00
|
|
|
import { toRaw, isRef, pauseTracking, resetTracking } from '@vue/reactivity'
|
2019-10-28 00:55:50 +00:00
|
|
|
import { callWithErrorHandling, ErrorCodes } from './errorHandling'
|
2019-08-30 14:36:30 +00:00
|
|
|
|
2019-10-08 13:26:09 +00:00
|
|
|
type ComponentVNode = VNode & {
|
2020-08-19 20:11:29 +00:00
|
|
|
type: ConcreteComponent
|
2019-10-08 13:26:09 +00:00
|
|
|
}
|
|
|
|
|
2019-10-15 03:15:36 +00:00
|
|
|
const stack: VNode[] = []
|
2019-08-30 14:36:30 +00:00
|
|
|
|
|
|
|
type TraceEntry = {
|
2019-10-08 13:26:09 +00:00
|
|
|
vnode: ComponentVNode
|
2019-08-30 14:36:30 +00:00
|
|
|
recurseCount: number
|
|
|
|
}
|
|
|
|
|
|
|
|
type ComponentTraceStack = TraceEntry[]
|
|
|
|
|
2019-11-02 03:04:28 +00:00
|
|
|
export function pushWarningContext(vnode: VNode) {
|
2019-08-30 14:36:30 +00:00
|
|
|
stack.push(vnode)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function popWarningContext() {
|
|
|
|
stack.pop()
|
|
|
|
}
|
|
|
|
|
|
|
|
export function warn(msg: string, ...args: any[]) {
|
2019-11-03 03:21:02 +00:00
|
|
|
// avoid props formatting or warn handler tracking deps that might be mutated
|
|
|
|
// during patch, leading to infinite recursion.
|
|
|
|
pauseTracking()
|
|
|
|
|
2019-09-03 22:11:04 +00:00
|
|
|
const instance = stack.length ? stack[stack.length - 1].component : null
|
|
|
|
const appWarnHandler = instance && instance.appContext.config.warnHandler
|
|
|
|
const trace = getComponentTrace()
|
|
|
|
|
|
|
|
if (appWarnHandler) {
|
2019-10-28 00:55:50 +00:00
|
|
|
callWithErrorHandling(
|
|
|
|
appWarnHandler,
|
|
|
|
instance,
|
|
|
|
ErrorCodes.APP_WARN_HANDLER,
|
|
|
|
[
|
|
|
|
msg + args.join(''),
|
2019-12-10 16:14:29 +00:00
|
|
|
instance && instance.proxy,
|
2019-11-03 03:21:02 +00:00
|
|
|
trace
|
2020-06-26 13:28:15 +00:00
|
|
|
.map(
|
|
|
|
({ vnode }) => `at <${formatComponentName(instance, vnode.type)}>`
|
|
|
|
)
|
2019-11-03 03:21:02 +00:00
|
|
|
.join('\n'),
|
|
|
|
trace
|
2019-10-28 00:55:50 +00:00
|
|
|
]
|
2019-09-03 22:11:04 +00:00
|
|
|
)
|
2019-08-30 14:36:30 +00:00
|
|
|
} else {
|
2019-11-03 03:21:02 +00:00
|
|
|
const warnArgs = [`[Vue warn]: ${msg}`, ...args]
|
2020-07-15 14:38:45 +00:00
|
|
|
/* istanbul ignore if */
|
2019-11-03 03:21:02 +00:00
|
|
|
if (
|
|
|
|
trace.length &&
|
|
|
|
// avoid spamming console during tests
|
2019-11-04 16:24:22 +00:00
|
|
|
!__TEST__
|
2019-11-03 03:21:02 +00:00
|
|
|
) {
|
|
|
|
warnArgs.push(`\n`, ...formatTrace(trace))
|
|
|
|
}
|
|
|
|
console.warn(...warnArgs)
|
2019-08-30 14:36:30 +00:00
|
|
|
}
|
2019-11-03 03:21:02 +00:00
|
|
|
|
2020-02-18 04:14:07 +00:00
|
|
|
resetTracking()
|
2019-08-30 14:36:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function getComponentTrace(): ComponentTraceStack {
|
|
|
|
let currentVNode: VNode | null = stack[stack.length - 1]
|
|
|
|
if (!currentVNode) {
|
|
|
|
return []
|
|
|
|
}
|
|
|
|
|
|
|
|
// we can't just use the stack because it will be incomplete during updates
|
|
|
|
// that did not start from the root. Re-construct the parent chain using
|
|
|
|
// instance parent pointers.
|
2019-10-05 14:48:54 +00:00
|
|
|
const normalizedStack: ComponentTraceStack = []
|
2019-08-30 14:36:30 +00:00
|
|
|
|
|
|
|
while (currentVNode) {
|
2019-10-05 14:48:54 +00:00
|
|
|
const last = normalizedStack[0]
|
2019-08-30 14:36:30 +00:00
|
|
|
if (last && last.vnode === currentVNode) {
|
|
|
|
last.recurseCount++
|
|
|
|
} else {
|
2019-10-05 14:48:54 +00:00
|
|
|
normalizedStack.push({
|
2019-11-02 03:04:28 +00:00
|
|
|
vnode: currentVNode as ComponentVNode,
|
2019-08-30 14:36:30 +00:00
|
|
|
recurseCount: 0
|
|
|
|
})
|
|
|
|
}
|
2020-04-29 18:41:06 +00:00
|
|
|
const parentInstance: ComponentInternalInstance | null =
|
|
|
|
currentVNode.component && currentVNode.component.parent
|
2019-08-30 14:36:30 +00:00
|
|
|
currentVNode = parentInstance && parentInstance.vnode
|
|
|
|
}
|
|
|
|
|
2019-10-05 14:48:54 +00:00
|
|
|
return normalizedStack
|
2019-08-30 14:36:30 +00:00
|
|
|
}
|
|
|
|
|
2020-07-15 14:38:45 +00:00
|
|
|
/* istanbul ignore next */
|
2019-11-03 03:21:02 +00:00
|
|
|
function formatTrace(trace: ComponentTraceStack): any[] {
|
|
|
|
const logs: any[] = []
|
2019-09-03 22:11:04 +00:00
|
|
|
trace.forEach((entry, i) => {
|
2019-11-03 03:21:02 +00:00
|
|
|
logs.push(...(i === 0 ? [] : [`\n`]), ...formatTraceEntry(entry))
|
2019-09-03 22:11:04 +00:00
|
|
|
})
|
|
|
|
return logs
|
|
|
|
}
|
|
|
|
|
2019-11-03 03:21:02 +00:00
|
|
|
function formatTraceEntry({ vnode, recurseCount }: TraceEntry): any[] {
|
2019-08-30 14:36:30 +00:00
|
|
|
const postfix =
|
|
|
|
recurseCount > 0 ? `... (${recurseCount} recursive calls)` : ``
|
2020-04-29 18:41:06 +00:00
|
|
|
const isRoot = vnode.component ? vnode.component.parent == null : false
|
2020-06-26 13:28:15 +00:00
|
|
|
const open = ` at <${formatComponentName(
|
|
|
|
vnode.component,
|
|
|
|
vnode.type,
|
|
|
|
isRoot
|
|
|
|
)}`
|
2019-08-30 14:36:30 +00:00
|
|
|
const close = `>` + postfix
|
|
|
|
return vnode.props
|
2020-04-20 20:06:51 +00:00
|
|
|
? [open, ...formatProps(vnode.props), close]
|
|
|
|
: [open + close]
|
2019-08-30 14:36:30 +00:00
|
|
|
}
|
|
|
|
|
2020-07-15 14:38:45 +00:00
|
|
|
/* istanbul ignore next */
|
2019-11-03 03:21:02 +00:00
|
|
|
function formatProps(props: Data): any[] {
|
|
|
|
const res: any[] = []
|
2019-11-24 03:17:46 +00:00
|
|
|
const keys = Object.keys(props)
|
|
|
|
keys.slice(0, 3).forEach(key => {
|
2019-11-03 03:21:02 +00:00
|
|
|
res.push(...formatProp(key, props[key]))
|
2019-11-24 03:17:46 +00:00
|
|
|
})
|
|
|
|
if (keys.length > 3) {
|
|
|
|
res.push(` ...`)
|
2019-08-30 14:36:30 +00:00
|
|
|
}
|
|
|
|
return res
|
2019-05-28 10:06:00 +00:00
|
|
|
}
|
2019-11-03 03:21:02 +00:00
|
|
|
|
|
|
|
function formatProp(key: string, value: unknown): any[]
|
|
|
|
function formatProp(key: string, value: unknown, raw: true): any
|
2020-07-15 14:38:45 +00:00
|
|
|
/* istanbul ignore next */
|
2019-11-03 03:21:02 +00:00
|
|
|
function formatProp(key: string, value: unknown, raw?: boolean): any {
|
|
|
|
if (isString(value)) {
|
|
|
|
value = JSON.stringify(value)
|
|
|
|
return raw ? value : [`${key}=${value}`]
|
2019-11-24 03:17:46 +00:00
|
|
|
} else if (
|
|
|
|
typeof value === 'number' ||
|
|
|
|
typeof value === 'boolean' ||
|
|
|
|
value == null
|
|
|
|
) {
|
2019-11-03 03:21:02 +00:00
|
|
|
return raw ? value : [`${key}=${value}`]
|
|
|
|
} else if (isRef(value)) {
|
|
|
|
value = formatProp(key, toRaw(value.value), true)
|
|
|
|
return raw ? value : [`${key}=Ref<`, value, `>`]
|
2019-11-24 03:17:46 +00:00
|
|
|
} else if (isFunction(value)) {
|
|
|
|
return [`${key}=fn${value.name ? `<${value.name}>` : ``}`]
|
2019-11-03 03:21:02 +00:00
|
|
|
} else {
|
|
|
|
value = toRaw(value)
|
|
|
|
return raw ? value : [`${key}=`, value]
|
|
|
|
}
|
|
|
|
}
|