Breadcrumbs
A headless component for creating responsive breadcrumb navigation with proper ARIA support.
Usage
The Breadcrumbs component provides a compound component pattern for building navigation trails. It uses createBreadcrumbs, createGroup, and createOverflow internally.
Anatomy
<script setup lang="ts">
import { Breadcrumbs } from '@vuetify/v0'
</script>
<template>
<Breadcrumbs.Root>
<Breadcrumbs.List>
<Breadcrumbs.Item>
<Breadcrumbs.Link />
</Breadcrumbs.Item>
<Breadcrumbs.Divider />
<Breadcrumbs.Ellipsis />
<Breadcrumbs.Divider />
<Breadcrumbs.Item>
<Breadcrumbs.Page />
</Breadcrumbs.Item>
</Breadcrumbs.List>
</Breadcrumbs.Root>
</template>Architecture
The Root component composes three internal systems: createBreadcrumbs for navigation state, createGroup for visibility tracking, and createOverflow for width measurement.
The Root creates three internal composables: createBreadcrumbs manages the navigation model, createGroup tracks item visibility, and createOverflow measures widths to determine how many items fit.
Examples
Recipes
Common patterns for integrating Breadcrumbs into your application.
Links and Current Page
Use Breadcrumbs.Link for navigable items and Breadcrumbs.Page for the current (last) item. Page automatically applies aria-current="page".
<template>
<Breadcrumbs.Item>
<Breadcrumbs.Link href="/products">Products</Breadcrumbs.Link>
</Breadcrumbs.Item>
<!-- Last item uses Page instead of Link -->
<Breadcrumbs.Item>
<Breadcrumbs.Page>Current</Breadcrumbs.Page>
</Breadcrumbs.Item>
</template>With Vue Router
Use the as prop to render Breadcrumbs.Link as a RouterLink:
<template>
<Breadcrumbs.Item>
<Breadcrumbs.Link :as="RouterLink" to="/products">
Products
</Breadcrumbs.Link>
</Breadcrumbs.Item>
</template>Slot Props
The Root exposes navigation state and methods through its default slot:
<template>
<Breadcrumbs.Root v-slot="{ isOverflowing, depth, isRoot, first, prev, select }">
<!-- Use navigation methods and overflow state -->
</Breadcrumbs.Root>
</template>Item Gap
The gap prop controls the pixel gap between items used when calculating overflow capacity (default: 8). Adjust it to match your CSS gap so the overflow calculation stays accurate:
<template>
<!-- CSS gap is 16px — tell the component so overflow math is correct -->
<Breadcrumbs.Root :gap="16" class="flex gap-4">
<Breadcrumbs.Item v-for="crumb in crumbs" :key="crumb.path">
<Breadcrumbs.Link :href="crumb.path">{{ crumb.label }}</Breadcrumbs.Link>
</Breadcrumbs.Item>
</Breadcrumbs.Root>
</template>Custom Ellipsis
Override the ellipsis globally on Root or per-instance:
<template>
<Breadcrumbs.Root ellipsis="[more]">
<!-- Ellipsis shows "[more]" instead of default -->
</Breadcrumbs.Root>
</template>Plugins
Breadcrumbs integrates with v0’s plugin system for internationalization.
Locale
The Root uses useLocale internally for the navigation landmark’s aria-label. Without any configuration, it defaults to "Breadcrumb".
Override with a prop — no plugin needed:
<template>
<Breadcrumbs.Root label="Fil d'Ariane">
<!-- ... -->
</Breadcrumbs.Root>
</template>Override with the locale plugin — for app-wide i18n:
import { createApp } from 'vue'
import { createLocalePlugin } from '@vuetify/v0'
import App from './App.vue'
const app = createApp(App)
app.use(
createLocalePlugin({
messages: {
en: { 'Breadcrumbs.label': 'Breadcrumb' },
fr: { 'Breadcrumbs.label': "Fil d'Ariane" },
},
})
)
app.mount('#app')The label prop takes priority over locale messages, so you can still override individual instances when needed.