Fallthrough Attributes Overwrite Explicit Attributes in Vue 3
Rule
In Vue 3, fallthrough attributes overwrite explicitly set attributes on the root element (except class and style which are merged). This is a breaking change from Vue 2. To preserve explicit attribute values, use inheritAttrs: false and manually bind $attrs before the explicit attribute.
Why This Matters
- Silent behavior change from Vue 2 to Vue 3
- Can cause unexpected attribute values in migrated codebases
- Only
classandstylemerge intelligently; other attributes are overwritten - Affects component composition patterns and wrapper components
Bad Code
<!-- Parent.vue -->
<template>
<Child msg="Passed from Parent" />
</template>
<!-- Child.vue - UNEXPECTED BEHAVIOR -->
<template>
<GrandChild msg="Set in Child" />
</template>
<!--
Vue 3 Result: GrandChild receives msg="Passed from Parent"
The fallthrough attribute OVERWRITES the explicit one!
Vue 2 Result: GrandChild receives msg="Set in Child"
The explicit attribute took precedence
-->Another common case with data attributes
<!-- Parent.vue -->
<template>
<Button data-testid="parent-button" />
</template>
<!-- Button.vue - WRONG: explicit data-testid is overwritten -->
<template>
<button data-testid="submit-btn">Submit</button>
</template>
<!-- Result: <button data-testid="parent-button">Submit</button> -->
<!-- The component's intended test ID is lost! -->Good Code
Option 1: Control attribute order with inheritAttrs: false
<!-- Child.vue - CORRECT: Control attribute precedence -->
<script setup>
defineOptions({
inheritAttrs: false
})
</script>
<template>
<!-- v-bind="$attrs" FIRST, then explicit attribute -->
<GrandChild v-bind="$attrs" msg="Set in Child" />
</template>
<!--
Result: GrandChild receives msg="Set in Child"
Explicit attribute overrides fallthrough because it comes last
-->Option 2: Exclude specific attrs from fallthrough
<script setup>
import { computed, useAttrs } from 'vue'
defineOptions({
inheritAttrs: false
})
const attrs = useAttrs()
// Filter out attributes you want to control explicitly
const filteredAttrs = computed(() => {
const { msg, ...rest } = attrs
return rest
})
</script>
<template>
<GrandChild v-bind="filteredAttrs" msg="Set in Child" />
</template>Option 3: For wrapper components, declare as prop
<!-- Button.vue - BEST: Declare attributes you need to control -->
<script setup>
const props = defineProps({
dataTestid: {
type: String,
default: 'submit-btn'
}
})
defineOptions({
inheritAttrs: false
})
</script>
<template>
<button :data-testid="dataTestid" v-bind="$attrs">
<slot />
</button>
</template>Class and Style Are Special
Unlike other attributes, class and style merge rather than overwrite:
<!-- Parent.vue -->
<template>
<Button class="large" style="color: red" />
</template>
<!-- Button.vue -->
<template>
<button class="btn" style="padding: 10px">Submit</button>
</template>
<!--
Result: <button class="btn large" style="padding: 10px; color: red">
Both classes and styles are MERGED, not overwritten
-->Vue 2 to Vue 3 Migration Checklist
When migrating components that rely on attribute precedence:
- Identify components that set explicit attributes on root elements
- Check if parent components pass the same attributes
- If explicit values should take precedence:
- Add
inheritAttrs: false - Use
v-bind="$attrs"before explicit attributes