expect(`App already provides property with key "bar".`).toHaveBeenWarned()
})
+ test('runWithContext', () => {
+ const app = createApp({
+ setup() {
+ provide('foo', 'should not be seen')
+ return () => h('div')
+ }
+ })
+ app.provide('foo', 1)
+
+ expect(app.runWithContext(() => inject('foo'))).toBe(1)
+
+ // ensure the context is restored
+ inject('foo')
+ expect('inject() can only be used inside setup').toHaveBeenWarned()
+ })
+
test('component', () => {
const Root = {
// local override
unmount(): void
provide<T>(key: InjectionKey<T> | string, value: T): this
+ /**
+ * Runs a function with the app as active instance. This allows using of `inject()` within the function to get access
+ * to variables provided via `app.provide()`.
+ *
+ * @param fn - function to run with the app as active instance
+ */
+ runWithContext<T>(fn: () => T): T
+
// internal, but we need to expose these for the server-renderer and devtools
_uid: number
_component: ConcreteComponent
context.provides[key as string | symbol] = value
return app
+ },
+
+ runWithContext(fn) {
+ currentApp = app
+ try {
+ return fn()
+ } finally {
+ currentApp = null
+ }
}
})
return app
}
}
+
+/**
+ * @internal Used to identify the current app when using `inject()` within
+ * `app.runWithContext()`.
+ */
+export let currentApp: App<unknown> | null = null
import { isFunction } from '@vue/shared'
import { currentInstance } from './component'
import { currentRenderingInstance } from './componentRenderContext'
+import { currentApp } from './apiCreateApp'
import { warn } from './warning'
export interface InjectionKey<T> extends Symbol {}
// fallback to `currentRenderingInstance` so that this can be called in
// a functional component
const instance = currentInstance || currentRenderingInstance
- if (instance) {
+
+ // also support looking up from app-level provides w/ `app.runWithContext()`
+ if (instance || currentApp) {
// #2400
// to support `app.use` plugins,
// fallback to appContext's `provides` if the instance is at root
- const provides =
- instance.parent == null
+ const provides = instance
+ ? instance.parent == null
? instance.vnode.appContext && instance.vnode.appContext.provides
: instance.parent.provides
+ : currentApp!._context.provides
if (provides && (key as string | symbol) in provides) {
// TS doesn't allow symbol as index type
return provides[key as string]
} else if (arguments.length > 1) {
return treatDefaultAsFactory && isFunction(defaultValue)
- ? defaultValue.call(instance.proxy)
+ ? defaultValue.call(instance && instance.proxy)
: defaultValue
} else if (__DEV__) {
warn(`injection "${String(key)}" not found.`)