2019-09-12 03:44:37 +00:00
|
|
|
import { VNode, VNodeChild, isVNode } from './vnode'
|
2020-02-18 04:14:07 +00:00
|
|
|
import {
|
2020-02-27 00:01:42 +00:00
|
|
|
reactive,
|
2020-02-18 04:14:07 +00:00
|
|
|
ReactiveEffect,
|
|
|
|
shallowReadonly,
|
|
|
|
pauseTracking,
|
|
|
|
resetTracking
|
|
|
|
} from '@vue/reactivity'
|
2019-09-06 16:58:31 +00:00
|
|
|
import {
|
|
|
|
PublicInstanceProxyHandlers,
|
2019-12-10 16:14:29 +00:00
|
|
|
ComponentPublicInstance,
|
|
|
|
runtimeCompiledRenderProxyHandlers
|
2019-10-02 14:03:43 +00:00
|
|
|
} from './componentProxy'
|
2020-01-24 03:23:10 +00:00
|
|
|
import { ComponentPropsOptions, resolveProps } from './componentProps'
|
|
|
|
import { Slots, resolveSlots } from './componentSlots'
|
2019-08-30 19:05:39 +00:00
|
|
|
import { warn } from './warning'
|
|
|
|
import {
|
2019-09-06 16:58:31 +00:00
|
|
|
ErrorCodes,
|
2019-08-30 19:05:39 +00:00
|
|
|
callWithErrorHandling,
|
|
|
|
callWithAsyncErrorHandling
|
|
|
|
} from './errorHandling'
|
2019-12-22 16:27:40 +00:00
|
|
|
import { AppContext, createAppContext, AppConfig } from './apiCreateApp'
|
2019-10-18 16:34:45 +00:00
|
|
|
import { Directive, validateDirectiveName } from './directives'
|
2019-10-02 14:03:43 +00:00
|
|
|
import { applyOptions, ComponentOptions } from './apiOptions'
|
2019-09-06 15:19:22 +00:00
|
|
|
import {
|
|
|
|
EMPTY_OBJ,
|
|
|
|
isFunction,
|
|
|
|
capitalize,
|
|
|
|
NOOP,
|
2019-10-14 19:36:30 +00:00
|
|
|
isObject,
|
2019-10-16 02:18:55 +00:00
|
|
|
NO,
|
2019-10-17 19:47:26 +00:00
|
|
|
makeMap,
|
2020-01-20 16:24:08 +00:00
|
|
|
isPromise,
|
2020-01-26 19:13:55 +00:00
|
|
|
isArray,
|
2020-02-14 06:36:42 +00:00
|
|
|
hyphenate,
|
|
|
|
ShapeFlags
|
2019-09-06 15:19:22 +00:00
|
|
|
} from '@vue/shared'
|
2019-11-04 23:38:55 +00:00
|
|
|
import { SuspenseBoundary } from './components/Suspense'
|
2019-12-11 15:25:34 +00:00
|
|
|
import { CompilerOptions } from '@vue/compiler-core'
|
2020-01-16 22:45:08 +00:00
|
|
|
import {
|
|
|
|
currentRenderingInstance,
|
|
|
|
markAttrsAccessed
|
|
|
|
} from './componentRenderUtils'
|
2019-05-28 05:27:31 +00:00
|
|
|
|
2019-08-13 15:18:23 +00:00
|
|
|
export type Data = { [key: string]: unknown }
|
2019-05-29 02:43:27 +00:00
|
|
|
|
2019-12-16 15:36:48 +00:00
|
|
|
export interface SFCInternalOptions {
|
|
|
|
__scopeId?: string
|
2019-12-18 02:28:24 +00:00
|
|
|
__cssModules?: Data
|
2019-12-16 15:36:48 +00:00
|
|
|
__hmrId?: string
|
|
|
|
__hmrUpdated?: boolean
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface FunctionalComponent<P = {}> extends SFCInternalOptions {
|
2019-06-19 08:43:34 +00:00
|
|
|
(props: P, ctx: SetupContext): VNodeChild
|
2019-05-28 10:06:00 +00:00
|
|
|
props?: ComponentPropsOptions<P>
|
2019-10-25 16:12:17 +00:00
|
|
|
inheritAttrs?: boolean
|
2019-05-28 10:06:00 +00:00
|
|
|
displayName?: string
|
|
|
|
}
|
|
|
|
|
2020-03-12 15:46:32 +00:00
|
|
|
export interface ClassComponent {
|
|
|
|
new (...args: any[]): ComponentPublicInstance<any, any, any, any, any>
|
|
|
|
__vccOpts: ComponentOptions
|
|
|
|
}
|
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
export type Component = ComponentOptions | FunctionalComponent
|
2020-02-16 02:04:29 +00:00
|
|
|
|
|
|
|
// A type used in public APIs where a component type is expected.
|
|
|
|
// The constructor type is an artificial type returned by defineComponent().
|
|
|
|
export type PublicAPIComponent =
|
|
|
|
| Component
|
2020-03-12 15:46:32 +00:00
|
|
|
| { new (...args: any[]): ComponentPublicInstance<any, any, any, any, any> }
|
2020-02-16 02:04:29 +00:00
|
|
|
|
2019-10-30 02:28:38 +00:00
|
|
|
export { ComponentOptions }
|
2019-09-02 20:09:34 +00:00
|
|
|
|
2019-05-28 11:36:15 +00:00
|
|
|
type LifecycleHook = Function[] | null
|
|
|
|
|
2019-08-30 16:16:09 +00:00
|
|
|
export const enum LifecycleHooks {
|
|
|
|
BEFORE_CREATE = 'bc',
|
|
|
|
CREATED = 'c',
|
|
|
|
BEFORE_MOUNT = 'bm',
|
|
|
|
MOUNTED = 'm',
|
|
|
|
BEFORE_UPDATE = 'bu',
|
|
|
|
UPDATED = 'u',
|
|
|
|
BEFORE_UNMOUNT = 'bum',
|
|
|
|
UNMOUNTED = 'um',
|
|
|
|
DEACTIVATED = 'da',
|
|
|
|
ACTIVATED = 'a',
|
|
|
|
RENDER_TRIGGERED = 'rtg',
|
|
|
|
RENDER_TRACKED = 'rtc',
|
|
|
|
ERROR_CAPTURED = 'ec'
|
2019-05-28 11:36:15 +00:00
|
|
|
}
|
|
|
|
|
2020-01-20 16:24:08 +00:00
|
|
|
export type Emit = (event: string, ...args: unknown[]) => any[]
|
2019-09-06 00:36:35 +00:00
|
|
|
|
2019-09-06 15:19:22 +00:00
|
|
|
export interface SetupContext {
|
2019-06-19 08:43:34 +00:00
|
|
|
attrs: Data
|
|
|
|
slots: Slots
|
2019-09-06 00:36:35 +00:00
|
|
|
emit: Emit
|
2019-06-19 08:43:34 +00:00
|
|
|
}
|
|
|
|
|
2019-12-10 16:14:29 +00:00
|
|
|
export type RenderFunction = {
|
|
|
|
(): VNodeChild
|
2020-03-11 20:39:26 +00:00
|
|
|
_rc?: boolean // isRuntimeCompiled
|
2019-12-10 16:14:29 +00:00
|
|
|
}
|
2019-09-05 22:48:49 +00:00
|
|
|
|
2019-09-06 16:58:31 +00:00
|
|
|
export interface ComponentInternalInstance {
|
2020-03-16 22:41:11 +00:00
|
|
|
type: Component
|
2019-09-06 16:58:31 +00:00
|
|
|
parent: ComponentInternalInstance | null
|
2019-09-02 20:09:34 +00:00
|
|
|
appContext: AppContext
|
2019-09-06 16:58:31 +00:00
|
|
|
root: ComponentInternalInstance
|
2019-05-29 02:43:27 +00:00
|
|
|
vnode: VNode
|
2019-05-28 09:19:47 +00:00
|
|
|
next: VNode | null
|
2019-05-29 02:43:27 +00:00
|
|
|
subTree: VNode
|
2019-05-28 09:19:47 +00:00
|
|
|
update: ReactiveEffect
|
2019-09-05 22:48:49 +00:00
|
|
|
render: RenderFunction | null
|
2019-06-19 09:31:49 +00:00
|
|
|
effects: ReactiveEffect[] | null
|
2019-06-19 14:48:22 +00:00
|
|
|
provides: Data
|
2019-12-10 16:14:29 +00:00
|
|
|
// cache for proxy access type to avoid hasOwnProperty calls
|
2019-10-20 21:00:11 +00:00
|
|
|
accessCache: Data | null
|
|
|
|
// cache for render function values that rely on _ctx but won't need updates
|
|
|
|
// after initialized (e.g. inline handlers)
|
2020-02-18 19:43:04 +00:00
|
|
|
renderCache: (Function | VNode)[]
|
2019-06-19 08:43:34 +00:00
|
|
|
|
2019-10-30 02:28:38 +00:00
|
|
|
// assets for fast resolution
|
2019-09-04 15:36:27 +00:00
|
|
|
components: Record<string, Component>
|
|
|
|
directives: Record<string, Directive>
|
|
|
|
|
2019-05-29 02:43:27 +00:00
|
|
|
// the rest are only for stateful components
|
2019-09-06 00:36:35 +00:00
|
|
|
renderContext: Data
|
|
|
|
data: Data
|
|
|
|
props: Data
|
|
|
|
attrs: Data
|
|
|
|
slots: Slots
|
2019-12-10 16:14:29 +00:00
|
|
|
proxy: ComponentPublicInstance | null
|
|
|
|
// alternative proxy used only for runtime-compiled render functions using
|
|
|
|
// `with` block
|
|
|
|
withProxy: ComponentPublicInstance | null
|
2019-09-06 00:36:35 +00:00
|
|
|
propsProxy: Data | null
|
2019-06-19 08:43:34 +00:00
|
|
|
setupContext: SetupContext | null
|
2019-09-03 19:27:59 +00:00
|
|
|
refs: Data
|
2019-09-06 00:36:35 +00:00
|
|
|
emit: Emit
|
2019-08-21 13:50:20 +00:00
|
|
|
|
2019-10-30 02:28:38 +00:00
|
|
|
// suspense related
|
|
|
|
asyncDep: Promise<any> | null
|
|
|
|
asyncResolved: boolean
|
|
|
|
|
|
|
|
// storage for any extra properties
|
|
|
|
sink: { [key: string]: any }
|
2019-08-30 16:16:09 +00:00
|
|
|
|
|
|
|
// lifecycle
|
2019-11-23 04:32:53 +00:00
|
|
|
isMounted: boolean
|
2019-09-11 13:07:29 +00:00
|
|
|
isUnmounted: boolean
|
2019-10-30 02:28:38 +00:00
|
|
|
isDeactivated: boolean
|
2019-08-30 16:16:09 +00:00
|
|
|
[LifecycleHooks.BEFORE_CREATE]: LifecycleHook
|
|
|
|
[LifecycleHooks.CREATED]: LifecycleHook
|
|
|
|
[LifecycleHooks.BEFORE_MOUNT]: LifecycleHook
|
|
|
|
[LifecycleHooks.MOUNTED]: LifecycleHook
|
|
|
|
[LifecycleHooks.BEFORE_UPDATE]: LifecycleHook
|
|
|
|
[LifecycleHooks.UPDATED]: LifecycleHook
|
|
|
|
[LifecycleHooks.BEFORE_UNMOUNT]: LifecycleHook
|
|
|
|
[LifecycleHooks.UNMOUNTED]: LifecycleHook
|
|
|
|
[LifecycleHooks.RENDER_TRACKED]: LifecycleHook
|
|
|
|
[LifecycleHooks.RENDER_TRIGGERED]: LifecycleHook
|
|
|
|
[LifecycleHooks.ACTIVATED]: LifecycleHook
|
|
|
|
[LifecycleHooks.DEACTIVATED]: LifecycleHook
|
|
|
|
[LifecycleHooks.ERROR_CAPTURED]: LifecycleHook
|
2019-12-12 23:13:59 +00:00
|
|
|
|
|
|
|
// hmr marker (dev only)
|
|
|
|
renderUpdated?: boolean
|
2019-09-06 00:36:35 +00:00
|
|
|
}
|
2019-05-28 11:36:15 +00:00
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
const emptyAppContext = createAppContext()
|
|
|
|
|
2020-01-24 02:01:56 +00:00
|
|
|
export function createComponentInstance(
|
2019-08-28 16:13:36 +00:00
|
|
|
vnode: VNode,
|
2019-09-06 16:58:31 +00:00
|
|
|
parent: ComponentInternalInstance | null
|
2019-10-09 16:17:42 +00:00
|
|
|
) {
|
2019-09-03 22:11:04 +00:00
|
|
|
// inherit parent app context - or - if root, adopt from root vnode
|
|
|
|
const appContext =
|
|
|
|
(parent ? parent.appContext : vnode.appContext) || emptyAppContext
|
2019-10-09 16:17:42 +00:00
|
|
|
const instance: ComponentInternalInstance = {
|
2019-08-28 16:13:36 +00:00
|
|
|
vnode,
|
2019-06-03 01:43:28 +00:00
|
|
|
parent,
|
2019-09-03 22:11:04 +00:00
|
|
|
appContext,
|
2019-11-02 03:04:28 +00:00
|
|
|
type: vnode.type as Component,
|
2019-10-09 16:17:42 +00:00
|
|
|
root: null!, // set later so it can point to itself
|
2019-05-28 11:36:15 +00:00
|
|
|
next: null,
|
2019-10-09 16:17:42 +00:00
|
|
|
subTree: null!, // will be set synchronously right after creation
|
|
|
|
update: null!, // will be set synchronously right after creation
|
2019-06-12 07:43:19 +00:00
|
|
|
render: null,
|
2019-12-10 16:14:29 +00:00
|
|
|
proxy: null,
|
|
|
|
withProxy: null,
|
2019-05-30 15:16:15 +00:00
|
|
|
propsProxy: null,
|
2019-06-19 08:43:34 +00:00
|
|
|
setupContext: null,
|
2019-08-30 16:16:09 +00:00
|
|
|
effects: null,
|
2019-09-03 22:11:04 +00:00
|
|
|
provides: parent ? parent.provides : Object.create(appContext.provides),
|
2019-10-17 02:13:52 +00:00
|
|
|
accessCache: null!,
|
2020-02-18 19:43:04 +00:00
|
|
|
renderCache: [],
|
2019-05-28 11:36:15 +00:00
|
|
|
|
2019-08-30 16:16:09 +00:00
|
|
|
// setup context properties
|
2019-09-06 00:36:35 +00:00
|
|
|
renderContext: EMPTY_OBJ,
|
2019-08-30 16:16:09 +00:00
|
|
|
data: EMPTY_OBJ,
|
|
|
|
props: EMPTY_OBJ,
|
|
|
|
attrs: EMPTY_OBJ,
|
|
|
|
slots: EMPTY_OBJ,
|
|
|
|
refs: EMPTY_OBJ,
|
|
|
|
|
2019-09-04 15:36:27 +00:00
|
|
|
// per-instance asset storage (mutable during options resolution)
|
|
|
|
components: Object.create(appContext.components),
|
|
|
|
directives: Object.create(appContext.directives),
|
|
|
|
|
2019-09-09 20:00:50 +00:00
|
|
|
// async dependency management
|
|
|
|
asyncDep: null,
|
|
|
|
asyncResolved: false,
|
|
|
|
|
2019-08-30 16:16:09 +00:00
|
|
|
// user namespace for storing whatever the user assigns to `this`
|
2019-10-30 02:28:38 +00:00
|
|
|
// can also be used as a wildcard storage for ad-hoc injections internally
|
|
|
|
sink: {},
|
2019-08-30 16:16:09 +00:00
|
|
|
|
|
|
|
// lifecycle hooks
|
|
|
|
// not using enums here because it results in computed properties
|
2019-11-23 04:32:53 +00:00
|
|
|
isMounted: false,
|
2019-09-11 13:07:29 +00:00
|
|
|
isUnmounted: false,
|
2019-10-30 02:28:38 +00:00
|
|
|
isDeactivated: false,
|
2019-08-30 16:16:09 +00:00
|
|
|
bc: null,
|
|
|
|
c: null,
|
2019-05-28 11:36:15 +00:00
|
|
|
bm: null,
|
|
|
|
m: null,
|
|
|
|
bu: null,
|
|
|
|
u: null,
|
|
|
|
um: null,
|
|
|
|
bum: null,
|
|
|
|
da: null,
|
|
|
|
a: null,
|
|
|
|
rtg: null,
|
|
|
|
rtc: null,
|
|
|
|
ec: null,
|
2019-08-21 13:50:20 +00:00
|
|
|
|
2020-01-20 16:24:08 +00:00
|
|
|
emit: (event, ...args): any[] => {
|
2019-06-19 08:43:34 +00:00
|
|
|
const props = instance.vnode.props || EMPTY_OBJ
|
2020-01-26 19:13:55 +00:00
|
|
|
let handler = props[`on${event}`] || props[`on${capitalize(event)}`]
|
|
|
|
if (!handler && event.indexOf('update:') === 0) {
|
|
|
|
event = hyphenate(event)
|
|
|
|
handler = props[`on${event}`] || props[`on${capitalize(event)}`]
|
|
|
|
}
|
2019-06-19 08:43:34 +00:00
|
|
|
if (handler) {
|
2020-01-20 16:24:08 +00:00
|
|
|
const res = callWithAsyncErrorHandling(
|
2019-10-21 17:59:10 +00:00
|
|
|
handler,
|
|
|
|
instance,
|
|
|
|
ErrorCodes.COMPONENT_EVENT_HANDLER,
|
|
|
|
args
|
|
|
|
)
|
2020-01-20 16:24:08 +00:00
|
|
|
return isArray(res) ? res : [res]
|
|
|
|
} else {
|
|
|
|
return []
|
2019-06-19 08:43:34 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-28 11:36:15 +00:00
|
|
|
}
|
2019-06-03 01:43:28 +00:00
|
|
|
|
|
|
|
instance.root = parent ? parent.root : instance
|
|
|
|
return instance
|
2019-05-28 11:36:15 +00:00
|
|
|
}
|
|
|
|
|
2019-09-06 16:58:31 +00:00
|
|
|
export let currentInstance: ComponentInternalInstance | null = null
|
2019-09-11 14:09:00 +00:00
|
|
|
export let currentSuspense: SuspenseBoundary | null = null
|
2019-05-28 11:36:15 +00:00
|
|
|
|
2019-09-06 16:58:31 +00:00
|
|
|
export const getCurrentInstance: () => ComponentInternalInstance | null = () =>
|
2019-11-23 04:32:53 +00:00
|
|
|
currentInstance || currentRenderingInstance
|
2019-06-20 07:25:10 +00:00
|
|
|
|
2019-09-06 16:58:31 +00:00
|
|
|
export const setCurrentInstance = (
|
|
|
|
instance: ComponentInternalInstance | null
|
|
|
|
) => {
|
2019-08-30 16:16:09 +00:00
|
|
|
currentInstance = instance
|
|
|
|
}
|
|
|
|
|
2019-10-16 02:18:55 +00:00
|
|
|
const isBuiltInTag = /*#__PURE__*/ makeMap('slot,component')
|
2019-10-14 19:36:30 +00:00
|
|
|
|
|
|
|
export function validateComponentName(name: string, config: AppConfig) {
|
|
|
|
const appIsNativeTag = config.isNativeTag || NO
|
|
|
|
if (isBuiltInTag(name) || appIsNativeTag(name)) {
|
|
|
|
warn(
|
|
|
|
'Do not use built-in or reserved HTML elements as component id: ' + name
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-29 14:49:17 +00:00
|
|
|
export let isInSSRComponentSetup = false
|
|
|
|
|
2020-01-24 03:23:10 +00:00
|
|
|
export function setupComponent(
|
2019-09-11 14:09:00 +00:00
|
|
|
instance: ComponentInternalInstance,
|
2020-01-29 14:49:17 +00:00
|
|
|
parentSuspense: SuspenseBoundary | null,
|
|
|
|
isSSR = false
|
2019-09-11 14:09:00 +00:00
|
|
|
) {
|
2020-01-29 14:49:17 +00:00
|
|
|
isInSSRComponentSetup = isSSR
|
2020-01-24 03:23:10 +00:00
|
|
|
const propsOptions = instance.type.props
|
|
|
|
const { props, children, shapeFlag } = instance.vnode
|
|
|
|
resolveProps(instance, props, propsOptions)
|
|
|
|
resolveSlots(instance, children)
|
|
|
|
|
|
|
|
// setup stateful logic
|
2020-01-29 14:49:17 +00:00
|
|
|
let setupResult
|
2020-01-24 03:23:10 +00:00
|
|
|
if (shapeFlag & ShapeFlags.STATEFUL_COMPONENT) {
|
2020-03-06 19:52:15 +00:00
|
|
|
setupResult = setupStatefulComponent(instance, parentSuspense, isSSR)
|
2020-01-24 03:23:10 +00:00
|
|
|
}
|
2020-01-29 14:49:17 +00:00
|
|
|
isInSSRComponentSetup = false
|
|
|
|
return setupResult
|
2020-01-24 03:23:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function setupStatefulComponent(
|
|
|
|
instance: ComponentInternalInstance,
|
2020-03-06 19:52:15 +00:00
|
|
|
parentSuspense: SuspenseBoundary | null,
|
|
|
|
isSSR: boolean
|
2020-01-24 03:23:10 +00:00
|
|
|
) {
|
2019-05-29 02:43:27 +00:00
|
|
|
const Component = instance.type as ComponentOptions
|
2019-10-14 19:36:30 +00:00
|
|
|
|
|
|
|
if (__DEV__) {
|
|
|
|
if (Component.name) {
|
|
|
|
validateComponentName(Component.name, instance.appContext.config)
|
|
|
|
}
|
|
|
|
if (Component.components) {
|
|
|
|
const names = Object.keys(Component.components)
|
|
|
|
for (let i = 0; i < names.length; i++) {
|
2019-10-28 14:13:26 +00:00
|
|
|
validateComponentName(names[i], instance.appContext.config)
|
2019-10-14 19:36:30 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-18 16:34:45 +00:00
|
|
|
if (Component.directives) {
|
|
|
|
const names = Object.keys(Component.directives)
|
|
|
|
for (let i = 0; i < names.length; i++) {
|
|
|
|
validateDirectiveName(names[i])
|
|
|
|
}
|
|
|
|
}
|
2019-10-14 19:36:30 +00:00
|
|
|
}
|
2019-10-17 02:13:52 +00:00
|
|
|
// 0. create render proxy property access cache
|
2019-10-18 02:29:51 +00:00
|
|
|
instance.accessCache = {}
|
2019-12-10 16:14:29 +00:00
|
|
|
// 1. create public instance / render proxy
|
|
|
|
instance.proxy = new Proxy(instance, PublicInstanceProxyHandlers)
|
2019-09-03 19:27:59 +00:00
|
|
|
// 2. create props proxy
|
|
|
|
// the propsProxy is a reactive AND readonly proxy to the actual props.
|
|
|
|
// it will be updated in resolveProps() on updates before render
|
2020-03-10 19:28:13 +00:00
|
|
|
const propsProxy = (instance.propsProxy = isSSR
|
2020-01-24 16:39:52 +00:00
|
|
|
? instance.props
|
|
|
|
: shallowReadonly(instance.props))
|
2019-09-03 19:27:59 +00:00
|
|
|
// 3. call setup()
|
2019-05-30 15:16:15 +00:00
|
|
|
const { setup } = Component
|
|
|
|
if (setup) {
|
2019-06-19 08:43:34 +00:00
|
|
|
const setupContext = (instance.setupContext =
|
|
|
|
setup.length > 1 ? createSetupContext(instance) : null)
|
2019-09-09 20:00:50 +00:00
|
|
|
|
|
|
|
currentInstance = instance
|
2019-09-11 14:09:00 +00:00
|
|
|
currentSuspense = parentSuspense
|
2020-02-18 04:14:07 +00:00
|
|
|
pauseTracking()
|
2020-01-24 16:39:52 +00:00
|
|
|
const setupResult = callWithErrorHandling(
|
2019-08-30 19:05:39 +00:00
|
|
|
setup,
|
|
|
|
instance,
|
2019-09-06 16:58:31 +00:00
|
|
|
ErrorCodes.SETUP_FUNCTION,
|
2019-08-30 19:05:39 +00:00
|
|
|
[propsProxy, setupContext]
|
|
|
|
)
|
2020-02-18 04:14:07 +00:00
|
|
|
resetTracking()
|
2019-09-09 20:00:50 +00:00
|
|
|
currentInstance = null
|
2019-09-11 14:09:00 +00:00
|
|
|
currentSuspense = null
|
2019-06-19 14:48:22 +00:00
|
|
|
|
2019-10-17 19:47:26 +00:00
|
|
|
if (isPromise(setupResult)) {
|
2020-03-10 19:28:13 +00:00
|
|
|
if (isSSR) {
|
2020-01-24 16:39:52 +00:00
|
|
|
// return the promise so server-renderer can wait on it
|
2020-03-13 02:19:41 +00:00
|
|
|
return setupResult.then((resolvedResult: unknown) => {
|
2020-03-06 19:52:15 +00:00
|
|
|
handleSetupResult(instance, resolvedResult, parentSuspense, isSSR)
|
2020-01-26 21:13:12 +00:00
|
|
|
})
|
2020-01-24 16:39:52 +00:00
|
|
|
} else if (__FEATURE_SUSPENSE__) {
|
2019-09-09 20:28:32 +00:00
|
|
|
// async setup returned Promise.
|
|
|
|
// bail here and wait for re-entry.
|
2019-10-05 14:09:34 +00:00
|
|
|
instance.asyncDep = setupResult
|
2019-09-09 20:28:32 +00:00
|
|
|
} else if (__DEV__) {
|
|
|
|
warn(
|
|
|
|
`setup() returned a Promise, but the version of Vue you are using ` +
|
|
|
|
`does not support it yet.`
|
|
|
|
)
|
|
|
|
}
|
2019-06-12 07:43:19 +00:00
|
|
|
} else {
|
2020-03-06 19:52:15 +00:00
|
|
|
handleSetupResult(instance, setupResult, parentSuspense, isSSR)
|
2019-06-12 07:43:19 +00:00
|
|
|
}
|
2019-08-21 21:05:14 +00:00
|
|
|
} else {
|
2020-03-06 19:52:15 +00:00
|
|
|
finishComponentSetup(instance, parentSuspense, isSSR)
|
2019-09-09 20:00:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function handleSetupResult(
|
|
|
|
instance: ComponentInternalInstance,
|
2019-09-11 14:09:00 +00:00
|
|
|
setupResult: unknown,
|
2020-03-06 19:52:15 +00:00
|
|
|
parentSuspense: SuspenseBoundary | null,
|
|
|
|
isSSR: boolean
|
2019-09-09 20:00:50 +00:00
|
|
|
) {
|
|
|
|
if (isFunction(setupResult)) {
|
|
|
|
// setup returned an inline render function
|
|
|
|
instance.render = setupResult as RenderFunction
|
|
|
|
} else if (isObject(setupResult)) {
|
2019-09-12 03:44:37 +00:00
|
|
|
if (__DEV__ && isVNode(setupResult)) {
|
|
|
|
warn(
|
|
|
|
`setup() should not return VNodes directly - ` +
|
|
|
|
`return a render function instead.`
|
|
|
|
)
|
|
|
|
}
|
2019-09-09 20:00:50 +00:00
|
|
|
// setup returned bindings.
|
|
|
|
// assuming a render function compiled from template is present.
|
2020-02-27 00:01:42 +00:00
|
|
|
instance.renderContext = reactive(setupResult)
|
2019-09-09 20:00:50 +00:00
|
|
|
} else if (__DEV__ && setupResult !== undefined) {
|
|
|
|
warn(
|
|
|
|
`setup() should return an object. Received: ${
|
|
|
|
setupResult === null ? 'null' : typeof setupResult
|
|
|
|
}`
|
|
|
|
)
|
|
|
|
}
|
2020-03-06 19:52:15 +00:00
|
|
|
finishComponentSetup(instance, parentSuspense, isSSR)
|
2019-09-09 20:00:50 +00:00
|
|
|
}
|
|
|
|
|
2019-09-20 16:16:19 +00:00
|
|
|
type CompileFunction = (
|
2019-12-11 15:25:34 +00:00
|
|
|
template: string | object,
|
2019-09-20 16:16:19 +00:00
|
|
|
options?: CompilerOptions
|
|
|
|
) => RenderFunction
|
|
|
|
|
|
|
|
let compile: CompileFunction | undefined
|
|
|
|
|
2019-11-02 02:54:01 +00:00
|
|
|
// exported method uses any to avoid d.ts relying on the compiler types.
|
|
|
|
export function registerRuntimeCompiler(_compile: any) {
|
2019-09-20 04:24:16 +00:00
|
|
|
compile = _compile
|
|
|
|
}
|
|
|
|
|
2019-09-11 14:09:00 +00:00
|
|
|
function finishComponentSetup(
|
|
|
|
instance: ComponentInternalInstance,
|
2020-03-06 19:52:15 +00:00
|
|
|
parentSuspense: SuspenseBoundary | null,
|
|
|
|
isSSR: boolean
|
2019-09-11 14:09:00 +00:00
|
|
|
) {
|
2019-09-09 20:00:50 +00:00
|
|
|
const Component = instance.type as ComponentOptions
|
2020-03-06 19:52:15 +00:00
|
|
|
|
|
|
|
// template / render function normalization
|
|
|
|
if (__NODE_JS__ && isSSR) {
|
|
|
|
if (Component.render) {
|
|
|
|
instance.render = Component.render as RenderFunction
|
|
|
|
}
|
|
|
|
} else if (!instance.render) {
|
2020-03-11 20:39:26 +00:00
|
|
|
if (compile && Component.template && !Component.render) {
|
|
|
|
Component.render = compile(Component.template, {
|
2019-12-11 15:25:34 +00:00
|
|
|
isCustomElement: instance.appContext.config.isCustomElement || NO
|
2019-10-15 21:50:38 +00:00
|
|
|
})
|
2019-12-15 03:15:38 +00:00
|
|
|
// mark the function as runtime compiled
|
2020-03-11 20:39:26 +00:00
|
|
|
;(Component.render as RenderFunction)._rc = true
|
2019-10-15 21:50:38 +00:00
|
|
|
}
|
2019-12-10 16:14:29 +00:00
|
|
|
|
2020-03-06 19:52:15 +00:00
|
|
|
if (__DEV__ && !Component.render) {
|
2019-10-15 21:50:38 +00:00
|
|
|
/* istanbul ignore if */
|
2020-03-11 20:39:26 +00:00
|
|
|
if (!compile && Component.template) {
|
2019-09-20 04:24:16 +00:00
|
|
|
warn(
|
|
|
|
`Component provides template but the build of Vue you are running ` +
|
2019-12-11 15:25:34 +00:00
|
|
|
`does not support runtime template compilation. Either use the ` +
|
2019-09-20 04:24:16 +00:00
|
|
|
`full build or pre-compile the template using Vue CLI.`
|
|
|
|
)
|
2020-01-29 14:49:17 +00:00
|
|
|
} else {
|
2020-03-11 20:39:26 +00:00
|
|
|
warn(`Component is missing template or render function.`)
|
2019-09-20 04:24:16 +00:00
|
|
|
}
|
|
|
|
}
|
2019-12-10 16:14:29 +00:00
|
|
|
|
2019-09-09 20:00:50 +00:00
|
|
|
instance.render = (Component.render || NOOP) as RenderFunction
|
2019-12-10 16:14:29 +00:00
|
|
|
|
|
|
|
// for runtime-compiled render functions using `with` blocks, the render
|
|
|
|
// proxy used needs a different `has` handler which is more performant and
|
|
|
|
// also only allows a whitelist of globals to fallthrough.
|
2020-03-11 20:39:26 +00:00
|
|
|
if (instance.render._rc) {
|
2019-12-10 16:14:29 +00:00
|
|
|
instance.withProxy = new Proxy(
|
|
|
|
instance,
|
|
|
|
runtimeCompiledRenderProxyHandlers
|
|
|
|
)
|
|
|
|
}
|
2019-05-28 11:36:15 +00:00
|
|
|
}
|
2019-09-09 20:00:50 +00:00
|
|
|
|
2019-09-04 02:25:38 +00:00
|
|
|
// support for 2.x options
|
|
|
|
if (__FEATURE_OPTIONS__) {
|
2019-09-09 20:00:50 +00:00
|
|
|
currentInstance = instance
|
2019-09-11 14:09:00 +00:00
|
|
|
currentSuspense = parentSuspense
|
2019-09-04 15:36:27 +00:00
|
|
|
applyOptions(instance, Component)
|
2019-09-09 20:00:50 +00:00
|
|
|
currentInstance = null
|
2019-09-11 14:09:00 +00:00
|
|
|
currentSuspense = null
|
2019-09-04 02:25:38 +00:00
|
|
|
}
|
2019-05-28 11:36:15 +00:00
|
|
|
}
|
2019-05-28 09:19:47 +00:00
|
|
|
|
2019-08-23 02:07:51 +00:00
|
|
|
// used to identify a setup context proxy
|
|
|
|
export const SetupProxySymbol = Symbol()
|
|
|
|
|
2019-06-19 08:43:34 +00:00
|
|
|
const SetupProxyHandlers: { [key: string]: ProxyHandler<any> } = {}
|
2019-10-21 14:36:38 +00:00
|
|
|
;['attrs', 'slots'].forEach((type: string) => {
|
2019-06-19 08:43:34 +00:00
|
|
|
SetupProxyHandlers[type] = {
|
2020-01-16 22:45:08 +00:00
|
|
|
get: (instance, key) => {
|
|
|
|
if (__DEV__) {
|
|
|
|
markAttrsAccessed()
|
|
|
|
}
|
2020-03-16 16:20:52 +00:00
|
|
|
// if the user pass the slots proxy to h(), normalizeChildren should not
|
|
|
|
// attempt to attach ctx to the object
|
|
|
|
if (key === '_') return 1
|
2020-01-16 22:45:08 +00:00
|
|
|
return instance[type][key]
|
|
|
|
},
|
2019-10-05 14:09:34 +00:00
|
|
|
has: (instance, key) => key === SetupProxySymbol || key in instance[type],
|
|
|
|
ownKeys: instance => Reflect.ownKeys(instance[type]),
|
2019-08-23 02:07:51 +00:00
|
|
|
// this is necessary for ownKeys to work properly
|
|
|
|
getOwnPropertyDescriptor: (instance, key) =>
|
|
|
|
Reflect.getOwnPropertyDescriptor(instance[type], key),
|
2019-06-19 08:43:34 +00:00
|
|
|
set: () => false,
|
|
|
|
deleteProperty: () => false
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2019-09-06 16:58:31 +00:00
|
|
|
function createSetupContext(instance: ComponentInternalInstance): SetupContext {
|
2019-06-19 08:43:34 +00:00
|
|
|
const context = {
|
2019-10-21 14:36:38 +00:00
|
|
|
// attrs & slots are non-reactive, but they need to always expose
|
2019-06-19 08:43:34 +00:00
|
|
|
// the latest values (instance.xxx may get replaced during updates) so we
|
|
|
|
// need to expose them through a proxy
|
|
|
|
attrs: new Proxy(instance, SetupProxyHandlers.attrs),
|
|
|
|
slots: new Proxy(instance, SetupProxyHandlers.slots),
|
2020-02-10 02:47:16 +00:00
|
|
|
get emit() {
|
|
|
|
return instance.emit
|
|
|
|
}
|
2019-10-05 14:09:34 +00:00
|
|
|
}
|
2019-06-19 08:43:34 +00:00
|
|
|
return __DEV__ ? Object.freeze(context) : context
|
|
|
|
}
|
2020-02-14 05:13:54 +00:00
|
|
|
|
|
|
|
// record effects created during a component's setup() so that they can be
|
|
|
|
// stopped when the component unmounts
|
|
|
|
export function recordInstanceBoundEffect(effect: ReactiveEffect) {
|
|
|
|
if (currentInstance) {
|
|
|
|
;(currentInstance.effects || (currentInstance.effects = [])).push(effect)
|
|
|
|
}
|
|
|
|
}
|