Skip to content

Commit ea5d0f3

Browse files
committed
fix(inject): fix edge case of provided with async-mutated getters
fix #12667
1 parent 25ffdb6 commit ea5d0f3

File tree

3 files changed

+63
-18
lines changed

3 files changed

+63
-18
lines changed

src/core/instance/inject.ts

+10-5
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,7 @@
11
import { warn, hasSymbol, isFunction, isObject } from '../util/index'
22
import { defineReactive, toggleObserving } from '../observer/index'
33
import type { Component } from 'types/component'
4-
import { provide } from 'v3/apiInject'
5-
import { setCurrentInstance } from '../../v3/currentInstance'
4+
import { resolveProvided } from 'v3/apiInject'
65

76
export function initProvide(vm: Component) {
87
const provideOption = vm.$options.provide
@@ -13,12 +12,18 @@ export function initProvide(vm: Component) {
1312
if (!isObject(provided)) {
1413
return
1514
}
15+
const source = resolveProvided(vm)
16+
// IE9 doesn't support Object.getOwnPropertyDescriptors so we have to
17+
// iterate the keys ourselves.
1618
const keys = hasSymbol ? Reflect.ownKeys(provided) : Object.keys(provided)
17-
setCurrentInstance(vm)
1819
for (let i = 0; i < keys.length; i++) {
19-
provide(keys[i], provided[keys[i]])
20+
const key = keys[i]
21+
Object.defineProperty(
22+
source,
23+
key,
24+
Object.getOwnPropertyDescriptor(provided, key)!
25+
)
2026
}
21-
setCurrentInstance()
2227
}
2328
}
2429

src/v3/apiInject.ts

+17-12
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import { isFunction, warn } from 'core/util'
22
import { currentInstance } from './currentInstance'
3+
import type { Component } from 'types/component'
34

45
export interface InjectionKey<T> extends Symbol {}
56

@@ -9,19 +10,23 @@ export function provide<T>(key: InjectionKey<T> | string | number, value: T) {
910
warn(`provide() can only be used inside setup().`)
1011
}
1112
} else {
12-
let provides = currentInstance._provided
13-
// by default an instance inherits its parent's provides object
14-
// but when it needs to provide values of its own, it creates its
15-
// own provides object using parent provides object as prototype.
16-
// this way in `inject` we can simply look up injections from direct
17-
// parent and let the prototype chain do the work.
18-
const parentProvides =
19-
currentInstance.$parent && currentInstance.$parent._provided
20-
if (parentProvides === provides) {
21-
provides = currentInstance._provided = Object.create(parentProvides)
22-
}
2313
// TS doesn't allow symbol as index type
24-
provides[key as string] = value
14+
resolveProvided(currentInstance)[key as string] = value
15+
}
16+
}
17+
18+
export function resolveProvided(vm: Component): Record<string, any> {
19+
// by default an instance inherits its parent's provides object
20+
// but when it needs to provide values of its own, it creates its
21+
// own provides object using parent provides object as prototype.
22+
// this way in `inject` we can simply look up injections from direct
23+
// parent and let the prototype chain do the work.
24+
const existing = vm._provided
25+
const parentProvides = vm.$parent && vm.$parent._provided
26+
if (parentProvides === existing) {
27+
return (vm._provided = Object.create(parentProvides))
28+
} else {
29+
return existing
2530
}
2631
}
2732

test/unit/features/options/inject.spec.ts

+36-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import Vue from 'vue'
22
import { Observer } from 'core/observer/index'
3-
import { isNative, isObject, hasOwn } from 'core/util/index'
3+
import { isNative, isObject, hasOwn, nextTick } from 'core/util/index'
44
import testObjectOption from '../../../helpers/test-object-option'
55

66
describe('Options provide/inject', () => {
@@ -677,4 +677,39 @@ describe('Options provide/inject', () => {
677677
})
678678
expect(`Injection "constructor" not found`).toHaveBeenWarned()
679679
})
680+
681+
// #12667
682+
test('provide with getters', async () => {
683+
const spy = vi.fn()
684+
const Child = {
685+
render() {},
686+
inject: ['foo'],
687+
mounted() {
688+
spy(this.foo)
689+
}
690+
}
691+
692+
let val = 1
693+
const vm = new Vue({
694+
components: { Child },
695+
template: `<Child v-if="ok" />`,
696+
data() {
697+
return {
698+
ok: false
699+
}
700+
},
701+
provide() {
702+
return {
703+
get foo() {
704+
return val
705+
}
706+
}
707+
}
708+
}).$mount()
709+
710+
val = 2
711+
vm.ok = true
712+
await nextTick()
713+
expect(spy).toHaveBeenCalledWith(2)
714+
})
680715
})

0 commit comments

Comments
 (0)