Skip to content

Commit 6df0e73

Browse files
authored
fix(runtime-core): properly call lifecycle hooks in chained mixins & extends (#1974)
fix #1973
1 parent 9153fc2 commit 6df0e73

File tree

2 files changed

+87
-4
lines changed

2 files changed

+87
-4
lines changed

packages/runtime-core/__tests__/apiOptions.spec.ts

+65
Original file line numberDiff line numberDiff line change
@@ -633,6 +633,71 @@ describe('api: options', () => {
633633
expect(calls).toEqual(['base', 'mixin', 'comp'])
634634
})
635635

636+
test('beforeCreate/created in extends and mixins', () => {
637+
const calls: string[] = []
638+
const BaseA = {
639+
beforeCreate() {
640+
calls.push('beforeCreateA')
641+
},
642+
created() {
643+
calls.push('createdA')
644+
}
645+
}
646+
const BaseB = {
647+
extends: BaseA,
648+
beforeCreate() {
649+
calls.push('beforeCreateB')
650+
},
651+
created() {
652+
calls.push('createdB')
653+
}
654+
}
655+
656+
const MixinA = {
657+
beforeCreate() {
658+
calls.push('beforeCreateC')
659+
},
660+
created() {
661+
calls.push('createdC')
662+
}
663+
}
664+
const MixinB = {
665+
mixins: [MixinA],
666+
beforeCreate() {
667+
calls.push('beforeCreateD')
668+
},
669+
created() {
670+
calls.push('createdD')
671+
}
672+
}
673+
674+
const Comp = {
675+
extends: BaseB,
676+
mixins: [MixinB],
677+
beforeCreate() {
678+
calls.push('selfBeforeCreate')
679+
},
680+
created() {
681+
calls.push('selfCreated')
682+
},
683+
render() {}
684+
}
685+
686+
renderToString(h(Comp))
687+
expect(calls).toEqual([
688+
'beforeCreateA',
689+
'beforeCreateB',
690+
'beforeCreateC',
691+
'beforeCreateD',
692+
'selfBeforeCreate',
693+
'createdA',
694+
'createdB',
695+
'createdC',
696+
'createdD',
697+
'selfCreated'
698+
])
699+
})
700+
636701
test('accessing setup() state from options', async () => {
637702
const Comp = defineComponent({
638703
setup() {

packages/runtime-core/src/componentOptions.ts

+22-4
Original file line numberDiff line numberDiff line change
@@ -614,11 +614,11 @@ function callSyncHook(
614614
globalMixins: ComponentOptions[]
615615
) {
616616
callHookFromMixins(name, globalMixins, ctx)
617-
const baseHook = options.extends && options.extends[name]
618-
if (baseHook) {
619-
baseHook.call(ctx)
617+
618+
const { extends: base, mixins } = options
619+
if (base) {
620+
callHookFromExtends(name, base, ctx)
620621
}
621-
const mixins = options.mixins
622622
if (mixins) {
623623
callHookFromMixins(name, mixins, ctx)
624624
}
@@ -628,12 +628,30 @@ function callSyncHook(
628628
}
629629
}
630630

631+
function callHookFromExtends(
632+
name: 'beforeCreate' | 'created',
633+
base: ComponentOptions,
634+
ctx: ComponentPublicInstance
635+
) {
636+
if (base.extends) {
637+
callHookFromExtends(name, base.extends, ctx)
638+
}
639+
const baseHook = base[name]
640+
if (baseHook) {
641+
baseHook.call(ctx)
642+
}
643+
}
644+
631645
function callHookFromMixins(
632646
name: 'beforeCreate' | 'created',
633647
mixins: ComponentOptions[],
634648
ctx: ComponentPublicInstance
635649
) {
636650
for (let i = 0; i < mixins.length; i++) {
651+
const chainedMixins = mixins[i].mixins
652+
if (chainedMixins) {
653+
callHookFromMixins(name, chainedMixins, ctx)
654+
}
637655
const fn = mixins[i][name]
638656
if (fn) {
639657
fn.call(ctx)

0 commit comments

Comments
 (0)