2020-02-16 02:04:29 +00:00
|
|
|
import {
|
|
|
|
Component,
|
|
|
|
Data,
|
|
|
|
validateComponentName,
|
|
|
|
PublicAPIComponent
|
|
|
|
} from './component'
|
2020-04-03 23:08:17 +00:00
|
|
|
import { ComponentOptions } from './componentOptions'
|
2019-10-02 14:03:43 +00:00
|
|
|
import { ComponentPublicInstance } from './componentProxy'
|
2019-10-18 16:34:45 +00:00
|
|
|
import { Directive, validateDirectiveName } from './directives'
|
2019-11-02 16:18:35 +00:00
|
|
|
import { RootRenderFunction } from './renderer'
|
2019-09-02 20:09:34 +00:00
|
|
|
import { InjectionKey } from './apiInject'
|
2019-12-09 19:06:21 +00:00
|
|
|
import { isFunction, NO, isObject } from '@vue/shared'
|
2019-09-02 20:09:34 +00:00
|
|
|
import { warn } from './warning'
|
2020-02-14 04:31:03 +00:00
|
|
|
import { createVNode, cloneVNode, VNode } from './vnode'
|
2020-03-23 15:08:22 +00:00
|
|
|
import { RootHydrateFunction } from './hydration'
|
2020-07-21 01:51:30 +00:00
|
|
|
import { devtoolsInitApp, devtoolsUnmountApp } from './devtools'
|
2020-06-26 13:03:55 +00:00
|
|
|
import { version } from '.'
|
2019-09-02 20:09:34 +00:00
|
|
|
|
2019-09-06 20:58:32 +00:00
|
|
|
export interface App<HostElement = any> {
|
2020-06-26 13:03:55 +00:00
|
|
|
version: string
|
2019-09-02 20:09:34 +00:00
|
|
|
config: AppConfig
|
2019-12-24 15:33:47 +00:00
|
|
|
use(plugin: Plugin, ...options: any[]): this
|
2019-09-02 20:09:34 +00:00
|
|
|
mixin(mixin: ComponentOptions): this
|
2020-03-12 14:19:30 +00:00
|
|
|
component(name: string): PublicAPIComponent | undefined
|
|
|
|
component(name: string, component: PublicAPIComponent): this
|
2019-09-02 20:09:34 +00:00
|
|
|
directive(name: string): Directive | undefined
|
|
|
|
directive(name: string, directive: Directive): this
|
2020-02-14 04:31:03 +00:00
|
|
|
mount(
|
|
|
|
rootContainer: HostElement | string,
|
|
|
|
isHydrate?: boolean
|
|
|
|
): ComponentPublicInstance
|
2020-01-16 17:23:47 +00:00
|
|
|
unmount(rootContainer: HostElement | string): void
|
2019-10-28 00:54:33 +00:00
|
|
|
provide<T>(key: InjectionKey<T> | string, value: T): this
|
2020-01-24 02:01:56 +00:00
|
|
|
|
2020-07-21 01:51:30 +00:00
|
|
|
// internal, but we need to expose these for the server-renderer and devtools
|
2020-01-24 02:01:56 +00:00
|
|
|
_component: Component
|
|
|
|
_props: Data | null
|
|
|
|
_container: HostElement | null
|
2020-01-30 17:20:23 +00:00
|
|
|
_context: AppContext
|
2019-09-02 20:09:34 +00:00
|
|
|
}
|
|
|
|
|
2020-03-24 15:59:00 +00:00
|
|
|
export type OptionMergeFunction = (
|
|
|
|
to: unknown,
|
|
|
|
from: unknown,
|
|
|
|
instance: any,
|
|
|
|
key: string
|
|
|
|
) => any
|
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
export interface AppConfig {
|
2020-03-24 15:59:00 +00:00
|
|
|
// @private
|
|
|
|
readonly isNativeTag?: (tag: string) => boolean
|
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
performance: boolean
|
2020-03-24 15:59:00 +00:00
|
|
|
optionMergeStrategies: Record<string, OptionMergeFunction>
|
2020-03-25 13:28:37 +00:00
|
|
|
globalProperties: Record<string, any>
|
2020-03-24 15:59:00 +00:00
|
|
|
isCustomElement: (tag: string) => boolean
|
2019-09-02 20:09:34 +00:00
|
|
|
errorHandler?: (
|
2020-03-09 19:58:52 +00:00
|
|
|
err: unknown,
|
2019-09-06 16:58:31 +00:00
|
|
|
instance: ComponentPublicInstance | null,
|
2019-09-02 20:09:34 +00:00
|
|
|
info: string
|
|
|
|
) => void
|
|
|
|
warnHandler?: (
|
|
|
|
msg: string,
|
2019-09-06 16:58:31 +00:00
|
|
|
instance: ComponentPublicInstance | null,
|
2019-09-02 20:09:34 +00:00
|
|
|
trace: string
|
|
|
|
) => void
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface AppContext {
|
2020-07-21 01:51:30 +00:00
|
|
|
app: App // for devtools
|
2019-09-02 20:09:34 +00:00
|
|
|
config: AppConfig
|
|
|
|
mixins: ComponentOptions[]
|
2020-03-12 14:19:30 +00:00
|
|
|
components: Record<string, PublicAPIComponent>
|
2019-09-02 20:09:34 +00:00
|
|
|
directives: Record<string, Directive>
|
|
|
|
provides: Record<string | symbol, any>
|
2019-12-22 17:25:04 +00:00
|
|
|
reload?: () => void // HMR only
|
2019-09-02 20:09:34 +00:00
|
|
|
}
|
|
|
|
|
2019-12-24 15:33:47 +00:00
|
|
|
type PluginInstallFunction = (app: App, ...options: any[]) => any
|
2019-09-02 20:09:34 +00:00
|
|
|
|
2019-09-03 22:11:04 +00:00
|
|
|
export type Plugin =
|
2020-01-08 17:40:24 +00:00
|
|
|
| PluginInstallFunction & { install?: PluginInstallFunction }
|
2019-09-02 20:09:34 +00:00
|
|
|
| {
|
|
|
|
install: PluginInstallFunction
|
|
|
|
}
|
|
|
|
|
|
|
|
export function createAppContext(): AppContext {
|
|
|
|
return {
|
2020-07-21 01:51:30 +00:00
|
|
|
app: null as any,
|
2019-09-02 20:09:34 +00:00
|
|
|
config: {
|
2020-03-24 15:59:00 +00:00
|
|
|
isNativeTag: NO,
|
2019-09-02 20:09:34 +00:00
|
|
|
performance: false,
|
2020-03-25 13:28:37 +00:00
|
|
|
globalProperties: {},
|
2020-03-24 15:59:00 +00:00
|
|
|
optionMergeStrategies: {},
|
2019-10-15 21:30:47 +00:00
|
|
|
isCustomElement: NO,
|
2019-09-02 20:09:34 +00:00
|
|
|
errorHandler: undefined,
|
2019-09-02 20:43:26 +00:00
|
|
|
warnHandler: undefined
|
2019-09-02 20:09:34 +00:00
|
|
|
},
|
|
|
|
mixins: [],
|
|
|
|
components: {},
|
|
|
|
directives: {},
|
2020-02-26 15:20:10 +00:00
|
|
|
provides: Object.create(null)
|
2019-09-02 20:09:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-23 20:05:38 +00:00
|
|
|
export type CreateAppFunction<HostElement> = (
|
2020-02-16 02:04:29 +00:00
|
|
|
rootComponent: PublicAPIComponent,
|
2020-01-23 20:05:38 +00:00
|
|
|
rootProps?: Data | null
|
|
|
|
) => App<HostElement>
|
|
|
|
|
2020-03-23 15:08:22 +00:00
|
|
|
export function createAppAPI<HostElement>(
|
|
|
|
render: RootRenderFunction,
|
|
|
|
hydrate?: RootHydrateFunction
|
2020-01-23 20:05:38 +00:00
|
|
|
): CreateAppFunction<HostElement> {
|
2020-03-23 15:08:22 +00:00
|
|
|
return function createApp(rootComponent, rootProps = null) {
|
2020-01-24 02:01:56 +00:00
|
|
|
if (rootProps != null && !isObject(rootProps)) {
|
|
|
|
__DEV__ && warn(`root props passed to app.mount() must be an object.`)
|
|
|
|
rootProps = null
|
|
|
|
}
|
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
const context = createAppContext()
|
2019-11-05 15:49:00 +00:00
|
|
|
const installedPlugins = new Set()
|
2019-09-02 20:09:34 +00:00
|
|
|
|
2019-09-03 22:11:04 +00:00
|
|
|
let isMounted = false
|
|
|
|
|
2020-07-21 01:51:30 +00:00
|
|
|
const app: App = (context.app = {
|
2020-03-23 15:08:22 +00:00
|
|
|
_component: rootComponent as Component,
|
2020-01-24 02:01:56 +00:00
|
|
|
_props: rootProps,
|
|
|
|
_container: null,
|
2020-01-30 17:20:23 +00:00
|
|
|
_context: context,
|
2020-01-23 20:05:38 +00:00
|
|
|
|
2020-06-26 13:03:55 +00:00
|
|
|
version,
|
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
get config() {
|
|
|
|
return context.config
|
|
|
|
},
|
|
|
|
|
|
|
|
set config(v) {
|
2019-09-02 20:16:08 +00:00
|
|
|
if (__DEV__) {
|
|
|
|
warn(
|
|
|
|
`app.config cannot be replaced. Modify individual options instead.`
|
|
|
|
)
|
|
|
|
}
|
2019-09-02 20:09:34 +00:00
|
|
|
},
|
|
|
|
|
2019-12-24 15:33:47 +00:00
|
|
|
use(plugin: Plugin, ...options: any[]) {
|
2019-11-05 15:49:00 +00:00
|
|
|
if (installedPlugins.has(plugin)) {
|
|
|
|
__DEV__ && warn(`Plugin has already been applied to target app.`)
|
2019-11-27 14:18:03 +00:00
|
|
|
} else if (plugin && isFunction(plugin.install)) {
|
2019-11-05 15:49:00 +00:00
|
|
|
installedPlugins.add(plugin)
|
2019-12-24 15:33:47 +00:00
|
|
|
plugin.install(app, ...options)
|
2020-01-08 17:40:24 +00:00
|
|
|
} else if (isFunction(plugin)) {
|
|
|
|
installedPlugins.add(plugin)
|
|
|
|
plugin(app, ...options)
|
2019-09-02 20:09:34 +00:00
|
|
|
} else if (__DEV__) {
|
|
|
|
warn(
|
|
|
|
`A plugin must either be a function or an object with an "install" ` +
|
|
|
|
`function.`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return app
|
|
|
|
},
|
|
|
|
|
|
|
|
mixin(mixin: ComponentOptions) {
|
2020-07-21 01:51:30 +00:00
|
|
|
if (__FEATURE_OPTIONS_API__) {
|
2020-02-15 16:40:09 +00:00
|
|
|
if (!context.mixins.includes(mixin)) {
|
|
|
|
context.mixins.push(mixin)
|
|
|
|
} else if (__DEV__) {
|
|
|
|
warn(
|
|
|
|
'Mixin has already been applied to target app' +
|
|
|
|
(mixin.name ? `: ${mixin.name}` : '')
|
|
|
|
)
|
|
|
|
}
|
2019-10-28 20:22:03 +00:00
|
|
|
} else if (__DEV__) {
|
2020-02-15 16:40:09 +00:00
|
|
|
warn('Mixins are only available in builds supporting Options API')
|
2019-10-28 20:22:03 +00:00
|
|
|
}
|
2019-09-02 20:09:34 +00:00
|
|
|
return app
|
|
|
|
},
|
|
|
|
|
2020-02-16 02:04:29 +00:00
|
|
|
component(name: string, component?: PublicAPIComponent): any {
|
2019-10-14 19:36:30 +00:00
|
|
|
if (__DEV__) {
|
|
|
|
validateComponentName(name, context.config)
|
|
|
|
}
|
2019-09-02 20:09:34 +00:00
|
|
|
if (!component) {
|
2019-10-10 18:54:06 +00:00
|
|
|
return context.components[name]
|
2019-09-02 20:09:34 +00:00
|
|
|
}
|
2019-11-27 14:18:03 +00:00
|
|
|
if (__DEV__ && context.components[name]) {
|
|
|
|
warn(`Component "${name}" has already been registered in target app.`)
|
|
|
|
}
|
2020-03-12 14:19:30 +00:00
|
|
|
context.components[name] = component
|
2019-11-27 14:18:03 +00:00
|
|
|
return app
|
2019-09-02 20:09:34 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
directive(name: string, directive?: Directive) {
|
2019-10-18 16:34:45 +00:00
|
|
|
if (__DEV__) {
|
|
|
|
validateDirectiveName(name)
|
|
|
|
}
|
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
if (!directive) {
|
|
|
|
return context.directives[name] as any
|
|
|
|
}
|
2019-11-27 14:18:03 +00:00
|
|
|
if (__DEV__ && context.directives[name]) {
|
|
|
|
warn(`Directive "${name}" has already been registered in target app.`)
|
|
|
|
}
|
|
|
|
context.directives[name] = directive
|
|
|
|
return app
|
2019-09-02 20:09:34 +00:00
|
|
|
},
|
|
|
|
|
2020-02-14 04:31:03 +00:00
|
|
|
mount(rootContainer: HostElement, isHydrate?: boolean): any {
|
2019-09-03 22:11:04 +00:00
|
|
|
if (!isMounted) {
|
2020-03-23 15:08:22 +00:00
|
|
|
const vnode = createVNode(rootComponent as Component, rootProps)
|
2019-09-03 22:11:04 +00:00
|
|
|
// store app context on the root VNode.
|
|
|
|
// this will be set on the root instance on initial mount.
|
|
|
|
vnode.appContext = context
|
2019-12-22 17:25:04 +00:00
|
|
|
|
|
|
|
// HMR root reload
|
2020-04-20 04:34:53 +00:00
|
|
|
if (__DEV__) {
|
2019-12-22 17:25:04 +00:00
|
|
|
context.reload = () => {
|
|
|
|
render(cloneVNode(vnode), rootContainer)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-14 06:30:08 +00:00
|
|
|
if (isHydrate && hydrate) {
|
2020-03-23 15:08:22 +00:00
|
|
|
hydrate(vnode as VNode<Node, Element>, rootContainer as any)
|
2020-02-14 04:31:03 +00:00
|
|
|
} else {
|
|
|
|
render(vnode, rootContainer)
|
|
|
|
}
|
2019-09-03 22:11:04 +00:00
|
|
|
isMounted = true
|
2020-01-24 02:01:56 +00:00
|
|
|
app._container = rootContainer
|
2020-07-21 01:51:30 +00:00
|
|
|
// for devtools and telemetry
|
|
|
|
;(rootContainer as any).__vue_app__ = app
|
2020-07-16 22:18:52 +00:00
|
|
|
|
2020-07-21 01:51:30 +00:00
|
|
|
if (__DEV__ || __FEATURE_PROD_DEVTOOLS__) {
|
|
|
|
devtoolsInitApp(app, version)
|
|
|
|
}
|
2020-07-16 22:18:52 +00:00
|
|
|
|
2019-12-10 16:14:29 +00:00
|
|
|
return vnode.component!.proxy
|
2019-09-03 22:11:04 +00:00
|
|
|
} else if (__DEV__) {
|
|
|
|
warn(
|
2020-06-11 21:20:38 +00:00
|
|
|
`App has already been mounted.\n` +
|
|
|
|
`If you want to remount the same app, move your app creation logic ` +
|
|
|
|
`into a factory function and create fresh app instances for each ` +
|
|
|
|
`mount - e.g. \`const createMyApp = () => createApp(App)\``
|
2019-09-03 22:11:04 +00:00
|
|
|
)
|
|
|
|
}
|
2019-09-02 20:09:34 +00:00
|
|
|
},
|
|
|
|
|
2020-01-23 20:05:38 +00:00
|
|
|
unmount() {
|
|
|
|
if (isMounted) {
|
2020-02-15 16:33:22 +00:00
|
|
|
render(null, app._container)
|
2020-07-21 01:51:30 +00:00
|
|
|
devtoolsUnmountApp(app)
|
2020-01-23 20:05:38 +00:00
|
|
|
} else if (__DEV__) {
|
|
|
|
warn(`Cannot unmount an app that is not mounted.`)
|
|
|
|
}
|
2020-01-16 17:23:47 +00:00
|
|
|
},
|
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
provide(key, value) {
|
|
|
|
if (__DEV__ && key in context.provides) {
|
|
|
|
warn(
|
2020-05-02 14:26:32 +00:00
|
|
|
`App already provides property with key "${String(key)}". ` +
|
2019-09-02 20:09:34 +00:00
|
|
|
`It will be overwritten with the new value.`
|
|
|
|
)
|
|
|
|
}
|
2019-10-08 16:43:13 +00:00
|
|
|
// TypeScript doesn't allow symbols as index type
|
|
|
|
// https://github.com/Microsoft/TypeScript/issues/24587
|
|
|
|
context.provides[key as string] = value
|
2019-10-28 00:54:33 +00:00
|
|
|
|
|
|
|
return app
|
2019-09-02 20:09:34 +00:00
|
|
|
}
|
2020-07-21 01:51:30 +00:00
|
|
|
})
|
2020-07-16 22:18:52 +00:00
|
|
|
|
2019-09-02 20:09:34 +00:00
|
|
|
return app
|
|
|
|
}
|
|
|
|
}
|