fix(runtime-core): fix resolving inheritAttrs from mixins (#3742)

fix #3741
This commit is contained in:
edison 2021-05-28 09:53:41 +08:00 committed by GitHub
parent 9b2e894017
commit d6607c9864
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 65 additions and 21 deletions

View File

@ -301,6 +301,34 @@ describe('attribute fallthrough', () => {
expect(root.innerHTML).toMatch(`<div>1</div>`) expect(root.innerHTML).toMatch(`<div>1</div>`)
}) })
// #3741
it('should not fallthrough with inheritAttrs: false from mixins', () => {
const Parent = {
render() {
return h(Child, { foo: 1, class: 'parent' })
}
}
const mixin = {
inheritAttrs: false
}
const Child = defineComponent({
mixins: [mixin],
props: ['foo'],
render() {
return h('div', this.foo)
}
})
const root = document.createElement('div')
document.body.appendChild(root)
render(h(Parent), root)
// should not contain class
expect(root.innerHTML).toMatch(`<div>1</div>`)
})
it('explicit spreading with inheritAttrs: false', () => { it('explicit spreading with inheritAttrs: false', () => {
const Parent = { const Parent = {
render() { render() {

View File

@ -285,6 +285,12 @@ export interface ComponentInternalInstance {
*/ */
emitsOptions: ObjectEmitsOptions | null emitsOptions: ObjectEmitsOptions | null
/**
* resolved inheritAttrs options
* @internal
*/
inheritAttrs?: boolean
// the rest are only for stateful components --------------------------------- // the rest are only for stateful components ---------------------------------
// main proxy that serves as the public instance (`this`) // main proxy that serves as the public instance (`this`)
@ -469,6 +475,9 @@ export function createComponentInstance(
// props default value // props default value
propsDefaults: EMPTY_OBJ, propsDefaults: EMPTY_OBJ,
// inheritAttrs
inheritAttrs: type.inheritAttrs,
// state // state
ctx: EMPTY_OBJ, ctx: EMPTY_OBJ,
data: EMPTY_OBJ, data: EMPTY_OBJ,

View File

@ -567,17 +567,14 @@ export function applyOptions(
errorCaptured, errorCaptured,
serverPrefetch, serverPrefetch,
// public API // public API
expose expose,
inheritAttrs
} = options } = options
const publicThis = instance.proxy! const publicThis = instance.proxy!
const ctx = instance.ctx const ctx = instance.ctx
const globalMixins = instance.appContext.mixins const globalMixins = instance.appContext.mixins
if (asMixin && render && instance.render === NOOP) {
instance.render = render as InternalRenderFunction
}
// applyOptions is called non-as-mixin once per instance // applyOptions is called non-as-mixin once per instance
if (!asMixin) { if (!asMixin) {
shouldCacheAccess = false shouldCacheAccess = false
@ -755,17 +752,6 @@ export function applyOptions(
}) })
} }
// asset options.
// To reduce memory usage, only components with mixins or extends will have
// resolved asset registry attached to instance.
if (asMixin) {
resolveInstanceAssets(instance, options, COMPONENTS)
resolveInstanceAssets(instance, options, DIRECTIVES)
if (__COMPAT__ && isCompatEnabled(DeprecationTypes.FILTERS, instance)) {
resolveInstanceAssets(instance, options, FILTERS)
}
}
// lifecycle options // lifecycle options
if (!asMixin) { if (!asMixin) {
callSyncHook( callSyncHook(
@ -831,6 +817,27 @@ export function applyOptions(
warn(`The \`expose\` option is ignored when used in mixins.`) warn(`The \`expose\` option is ignored when used in mixins.`)
} }
} }
// options that are handled when creating the instance but also need to be
// applied from mixins
if (asMixin) {
if (render && instance.render === NOOP) {
instance.render = render as InternalRenderFunction
}
if (inheritAttrs != null && instance.type.inheritAttrs == null) {
instance.inheritAttrs = inheritAttrs
}
// asset options.
// To reduce memory usage, only components with mixins or extends will have
// resolved asset registry attached to instance.
resolveInstanceAssets(instance, options, COMPONENTS)
resolveInstanceAssets(instance, options, DIRECTIVES)
if (__COMPAT__ && isCompatEnabled(DeprecationTypes.FILTERS, instance)) {
resolveInstanceAssets(instance, options, FILTERS)
}
}
} }
function resolveInstanceAssets( function resolveInstanceAssets(

View File

@ -55,7 +55,8 @@ export function renderComponentRoot(
renderCache, renderCache,
data, data,
setupState, setupState,
ctx ctx,
inheritAttrs
} = instance } = instance
let result let result
@ -123,7 +124,7 @@ export function renderComponentRoot(
;[root, setRoot] = getChildRoot(result) ;[root, setRoot] = getChildRoot(result)
} }
if (fallthroughAttrs && Component.inheritAttrs !== false) { if (fallthroughAttrs && inheritAttrs !== false) {
const keys = Object.keys(fallthroughAttrs) const keys = Object.keys(fallthroughAttrs)
const { shapeFlag } = root const { shapeFlag } = root
if (keys.length) { if (keys.length) {
@ -190,7 +191,7 @@ export function renderComponentRoot(
) { ) {
const { class: cls, style } = vnode.props || {} const { class: cls, style } = vnode.props || {}
if (cls || style) { if (cls || style) {
if (__DEV__ && Component.inheritAttrs === false) { if (__DEV__ && inheritAttrs === false) {
warnDeprecation( warnDeprecation(
DeprecationTypes.INSTANCE_ATTRS_CLASS_STYLE, DeprecationTypes.INSTANCE_ATTRS_CLASS_STYLE,
instance, instance,

View File

@ -132,8 +132,7 @@ function renderComponentSubTree(
if (ssrRender) { if (ssrRender) {
// optimized // optimized
// resolve fallthrough attrs // resolve fallthrough attrs
let attrs = let attrs = instance.inheritAttrs !== false ? instance.attrs : undefined
instance.type.inheritAttrs !== false ? instance.attrs : undefined
let hasCloned = false let hasCloned = false
let cur = instance let cur = instance