Skip to content

Commit 306997e

Browse files
znckyyx990803
authored andcommitted
fix(core): add merge strategy for provide option (#6025)
Fix #6008
1 parent 254d85c commit 306997e

File tree

2 files changed

+126
-11
lines changed

2 files changed

+126
-11
lines changed

src/core/util/options.js

+25-11
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ function mergeData (to: Object, from: ?Object): Object {
6363
/**
6464
* Data
6565
*/
66-
strats.data = function (
66+
export function mergeDataOrFn (
6767
parentVal: any,
6868
childVal: any,
6969
vm?: Component
@@ -73,15 +73,6 @@ strats.data = function (
7373
if (!childVal) {
7474
return parentVal
7575
}
76-
if (typeof childVal !== 'function') {
77-
process.env.NODE_ENV !== 'production' && warn(
78-
'The "data" option should be a function ' +
79-
'that returns a per-instance value in component ' +
80-
'definitions.',
81-
vm
82-
)
83-
return parentVal
84-
}
8576
if (!parentVal) {
8677
return childVal
8778
}
@@ -92,7 +83,7 @@ strats.data = function (
9283
// it has to be a function to pass previous merges.
9384
return function mergedDataFn () {
9485
return mergeData(
95-
childVal.call(this),
86+
typeof childVal === 'function' ? childVal.call(this) : childVal,
9687
parentVal.call(this)
9788
)
9889
}
@@ -114,6 +105,28 @@ strats.data = function (
114105
}
115106
}
116107

108+
strats.data = function (
109+
parentVal: any,
110+
childVal: any,
111+
vm?: Component
112+
): ?Function {
113+
if (!vm) {
114+
if (childVal && typeof childVal !== 'function') {
115+
process.env.NODE_ENV !== 'production' && warn(
116+
'The "data" option should be a function ' +
117+
'that returns a per-instance value in component ' +
118+
'definitions.',
119+
vm
120+
)
121+
122+
return parentVal
123+
}
124+
return mergeDataOrFn.call(this, parentVal, childVal)
125+
}
126+
127+
return mergeDataOrFn(parentVal, childVal, vm)
128+
}
129+
117130
/**
118131
* Hooks and props are merged as arrays.
119132
*/
@@ -191,6 +204,7 @@ strats.computed = function (parentVal: ?Object, childVal: ?Object): ?Object {
191204
extend(ret, childVal)
192205
return ret
193206
}
207+
strats.provide = mergeDataOrFn
194208

195209
/**
196210
* Default strategy.

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

+101
Original file line numberDiff line numberDiff line change
@@ -298,4 +298,105 @@ describe('Options provide/inject', () => {
298298
expect(`Injection "bar" not found`).not.toHaveBeenWarned()
299299
expect(`Injection "baz" not found`).not.toHaveBeenWarned()
300300
})
301+
302+
// Github issue #6008
303+
it('should merge provide from mixins (objects)', () => {
304+
const mixinA = { provide: { foo: 'foo' }}
305+
const mixinB = { provide: { bar: 'bar' }}
306+
const child = {
307+
inject: ['foo', 'bar'],
308+
template: `<span/>`,
309+
created () {
310+
injected = [this.foo, this.bar]
311+
}
312+
}
313+
new Vue({
314+
mixins: [mixinA, mixinB],
315+
render (h) {
316+
return h(child)
317+
}
318+
}).$mount()
319+
320+
expect(injected).toEqual(['foo', 'bar'])
321+
})
322+
it('should merge provide from mixins (functions)', () => {
323+
const mixinA = { provide: () => ({ foo: 'foo' }) }
324+
const mixinB = { provide: () => ({ bar: 'bar' }) }
325+
const child = {
326+
inject: ['foo', 'bar'],
327+
template: `<span/>`,
328+
created () {
329+
injected = [this.foo, this.bar]
330+
}
331+
}
332+
new Vue({
333+
mixins: [mixinA, mixinB],
334+
render (h) {
335+
return h(child)
336+
}
337+
}).$mount()
338+
339+
expect(injected).toEqual(['foo', 'bar'])
340+
})
341+
it('should merge provide from mixins (mix of objects and functions)', () => {
342+
const mixinA = { provide: { foo: 'foo' }}
343+
const mixinB = { provide: () => ({ bar: 'bar' }) }
344+
const mixinC = { provide: { baz: 'baz' }}
345+
const mixinD = { provide: () => ({ bam: 'bam' }) }
346+
const child = {
347+
inject: ['foo', 'bar', 'baz', 'bam'],
348+
template: `<span/>`,
349+
created () {
350+
injected = [this.foo, this.bar, this.baz, this.bam]
351+
}
352+
}
353+
new Vue({
354+
mixins: [mixinA, mixinB, mixinC, mixinD],
355+
render (h) {
356+
return h(child)
357+
}
358+
}).$mount()
359+
360+
expect(injected).toEqual(['foo', 'bar', 'baz', 'bam'])
361+
})
362+
it('should merge provide from mixins and override existing keys', () => {
363+
const mixinA = { provide: { foo: 'foo' }}
364+
const mixinB = { provide: { foo: 'bar' }}
365+
const child = {
366+
inject: ['foo'],
367+
template: `<span/>`,
368+
created () {
369+
injected = [this.foo]
370+
}
371+
}
372+
new Vue({
373+
mixins: [mixinA, mixinB],
374+
render (h) {
375+
return h(child)
376+
}
377+
}).$mount()
378+
379+
expect(injected).toEqual(['bar'])
380+
})
381+
it('should merge provide when Vue.extend', () => {
382+
const mixinA = { provide: () => ({ foo: 'foo' }) }
383+
const child = {
384+
inject: ['foo', 'bar'],
385+
template: `<span/>`,
386+
created () {
387+
injected = [this.foo, this.bar]
388+
}
389+
}
390+
const Ctor = Vue.extend({
391+
mixins: [mixinA],
392+
provide: { bar: 'bar' },
393+
render (h) {
394+
return h(child)
395+
}
396+
})
397+
398+
new Ctor().$mount()
399+
400+
expect(injected).toEqual(['foo', 'bar'])
401+
})
301402
})

0 commit comments

Comments
 (0)