- Accordion
- Alert
- Alert Dialog
- Aspect Ratio
- Avatar
- Badge
- Breadcrumb
- Button
- Button Group
- Calendar
- Card
- Carousel
- Chart
- Checkbox
- Collapsible
- Combobox
- Command
- Context Menu
- Data Table
- Date Picker
- Dialog
- Drawer
- Dropdown Menu
- Empty
- Field
- Form
- Hover Card
- Input
- Input Group
- Input OTP
- Item
- Kbd
- Label
- Marker
- Menubar
- Native Select
- Navigation Menu
- Number Field
- Pagination
- Pin Input
- Popover
- Progress
- Radio Group
- Range Calendar
- Resizable
- Scroll Area
- Select
- Separator
- Sheet
- Sidebar
- Skeleton
- Slider
- Sonner
- Spinner
- Stepper
- Switch
- Table
- Tabs
- Tags Input
- Textarea
- Toast
- Toggle
- Toggle Group
- Tooltip
- Typography
<script setup lang="ts">
import { GitBranchIcon, SearchIcon } from '@lucide/vue'
import {
Marker,
MarkerContent,
MarkerIcon,
} from '@/components/ui/marker'
import { Spinner } from '@/components/ui/spinner'
</script>
<template>
<div class="flex w-full max-w-sm flex-col gap-8 py-12">
<Marker>
<MarkerIcon>
<GitBranchIcon />
</MarkerIcon>
<MarkerContent>Switched to a new branch</MarkerContent>
</Marker>
<Marker role="status">
<MarkerIcon>
<Spinner />
</MarkerIcon>
<MarkerContent class="shimmer">
Thinking...
</MarkerContent>
</Marker>
<Marker variant="separator">
<MarkerContent>Conversation compacted</MarkerContent>
</Marker>
<Marker>
<MarkerIcon>
<SearchIcon />
</MarkerIcon>
<MarkerContent>Explored 4 files</MarkerContent>
</Marker>
</div>
</template>Installation
pnpm dlx shadcn-vue@latest add marker
Usage
<script setup lang="ts">
import { Marker, MarkerContent, MarkerIcon, } from '@/components/ui/marker'
</script>
<template>
<Marker>
<MarkerIcon>
<CheckIcon />
</MarkerIcon>
<MarkerContent>Explored 4 files</MarkerContent>
</Marker>
</template>Composition
Use the following composition to build a marker:
Marker
├── MarkerIcon
└── MarkerContentFeatures
- Inline marker, bordered row, and labeled separator variants
- Decorative icon slot that is hidden from assistive tech
- Polymorphic root via
renderfor link and button markers - Pairs with the
shimmerutility for streaming status text - Customizable styling through the
classprop on every part
Examples
Variants
Use variant to switch between an inline marker, bordered row, and labeled separator.
<script setup lang="ts">
import {
Marker,
MarkerContent,
} from '@/components/ui/marker'
</script>
<template>
<div class="flex w-full max-w-sm flex-col gap-8 py-12">
<Marker>
<MarkerContent>A default marker for inline notes.</MarkerContent>
</Marker>
<Marker variant="separator">
<MarkerContent>A separator marker</MarkerContent>
</Marker>
<Marker variant="border">
<MarkerContent>A border marker for row boundaries.</MarkerContent>
</Marker>
</div>
</template>| Variant | Description |
|---|---|
default | An inline marker for status, notes, and actions. |
border | A default marker with a bottom border under the row. |
separator | A centered label with divider lines on each side. |
Status
Set role="status" and include a Spinner for streaming or in-progress markers so updates are announced.
<script setup lang="ts">
import {
Marker,
MarkerContent,
MarkerIcon,
} from '@/components/ui/marker'
import { Spinner } from '@/components/ui/spinner'
</script>
<template>
<div class="flex w-full max-w-sm flex-col gap-8 py-12">
<Marker role="status">
<MarkerIcon>
<Spinner />
</MarkerIcon>
<MarkerContent>Compacting conversation</MarkerContent>
</Marker>
<Marker variant="separator" role="status">
<MarkerIcon>
<Spinner />
</MarkerIcon>
<MarkerContent>Running tests</MarkerContent>
</Marker>
</div>
</template>Shimmer
Add the shimmer utility class to MarkerContent for an animated streaming-text effect. The utility ships with the shadcn package — see the shimmer docs for installation.
<script setup lang="ts">
import {
Marker,
MarkerContent,
} from '@/components/ui/marker'
</script>
<template>
<div class="flex w-full max-w-sm flex-col gap-8 py-12">
<Marker role="status">
<MarkerContent class="shimmer">
Thinking...
</MarkerContent>
</Marker>
<Marker variant="separator" role="status">
<MarkerContent class="shimmer">
Reading 4 files
</MarkerContent>
</Marker>
</div>
</template>Separator
Use the separator variant for labeled dividers, such as dates or section breaks, in a conversation.
<script setup lang="ts">
import {
Marker,
MarkerContent,
} from '@/components/ui/marker'
</script>
<template>
<div class="flex w-full max-w-sm flex-col gap-8 py-12">
<Marker variant="separator">
<MarkerContent>Today</MarkerContent>
</Marker>
<Marker variant="separator">
<MarkerContent>Worked for 42s</MarkerContent>
</Marker>
<Marker variant="separator">
<MarkerContent>Conversation compacted</MarkerContent>
</Marker>
</div>
</template>Border
Use the border variant for status rows that should keep the default marker alignment while separating the next row.
<script setup lang="ts">
import { FileTextIcon, GitBranchIcon, SearchIcon } from '@lucide/vue'
import {
Marker,
MarkerContent,
MarkerIcon,
} from '@/components/ui/marker'
</script>
<template>
<div class="flex w-full max-w-sm flex-col gap-3 py-12">
<Marker variant="border">
<MarkerIcon>
<GitBranchIcon />
</MarkerIcon>
<MarkerContent>Switched to release-candidate</MarkerContent>
</Marker>
<Marker variant="border">
<MarkerIcon>
<SearchIcon />
</MarkerIcon>
<MarkerContent>Reviewed 8 related files</MarkerContent>
</Marker>
<Marker variant="border">
<MarkerIcon>
<FileTextIcon />
</MarkerIcon>
<MarkerContent>Opened implementation notes</MarkerContent>
</Marker>
</div>
</template>With Icon
Use MarkerIcon to render an icon alongside the content. Use flex-col to stack the icon above the content.
<script setup lang="ts">
import { BookOpenCheck, GitBranchIcon, SearchIcon } from '@lucide/vue'
import {
Marker,
MarkerContent,
MarkerIcon,
} from '@/components/ui/marker'
</script>
<template>
<div class="flex w-full max-w-sm flex-col gap-12 py-12">
<Marker>
<MarkerIcon>
<GitBranchIcon />
</MarkerIcon>
<MarkerContent>Switched to a new branch</MarkerContent>
</Marker>
<Marker variant="separator">
<MarkerIcon>
<SearchIcon />
</MarkerIcon>
<MarkerContent>Explored 4 files</MarkerContent>
</Marker>
<Marker class="flex-col">
<MarkerIcon>
<BookOpenCheck />
</MarkerIcon>
<MarkerContent>Syncing completed</MarkerContent>
</Marker>
</div>
</template>Links and Buttons
Turn a marker into a link or button with the render prop on Marker.
<script setup lang="ts">
import { GitBranchIcon, RotateCcwIcon } from '@lucide/vue'
import { toast } from 'vue-sonner'
import {
Marker,
MarkerContent,
MarkerIcon,
} from '@/components/ui/marker'
</script>
<template>
<div class="flex w-full max-w-sm flex-col gap-8 py-12">
<Marker as-child>
<a href="#links-and-buttons">
<MarkerIcon>
<GitBranchIcon />
</MarkerIcon>
<MarkerContent>View the pull request</MarkerContent>
</a>
</Marker>
<Marker as-child>
<button
type="button"
class="transition-colors hover:text-foreground"
@click="() => toast('You clicked the revert button')"
>
<MarkerIcon>
<RotateCcwIcon />
</MarkerIcon>
<MarkerContent>Revert this change</MarkerContent>
</button>
</Marker>
</div>
</template>import { Marker, MarkerContent } from "@/components/ui/marker"
<template>
<Marker as-child>
<a href="#links-and-buttons">
<MarkerIcon>
<GitBranchIcon />
</MarkerIcon>
<MarkerContent>View the pull request</MarkerContent>
</a>
</Marker>
</template>Accessibility
Marker is presentational by default. The correct semantics depend on how you use it, so choose the role based on intent rather than relying on a single default.
Status and Progress
For streaming or progress markers such as "Thinking..." or a running tool, set role="status" so assistive tech announces the update as it appears. Marker forwards role to the underlying element.
<Marker role="status">
<MarkerIcon>
<Spinner />
</MarkerIcon>
<MarkerContent>Compacting conversation</MarkerContent>
</Marker>Labeled Separators
A separator that carries text, such as a date or a section label, needs no role. The divider lines are decorative CSS pseudo-elements, and the text is announced as ordinary content.
<Marker variant="separator">
<MarkerContent>Today</MarkerContent>
</Marker>Note: Do not add role="separator" to a labeled divider. A separator
takes its accessible name from aria-label, not from its text, and its
contents are treated as presentational, so the visible label would not be
announced. Reserve role="separator" for a divider with no meaningful text.
Bordered Markers
A bordered marker keeps the same semantics as the default marker. The bottom border is decorative, so choose role="status", render, or no role based on the marker's purpose.
<Marker variant="border">
<MarkerIcon>
<FileTextIcon />
</MarkerIcon>
<MarkerContent>Opened implementation notes</MarkerContent>
</Marker>Decorative Icons
MarkerIcon is decorative and hidden from assistive tech with aria-hidden, so the adjacent MarkerContent carries the meaning. For an icon-only marker, provide an aria-label or visible text so it is not announced as empty.
<Marker aria-label="Synced">
<MarkerIcon>
<CheckIcon />
</MarkerIcon>
</Marker>Interactive Markers
When a marker links or triggers an action, render it as a real <button> or <a> with the render prop so it is focusable and exposes the correct role. The accessible name comes from the marker text.
<Marker as-child>
<a href="/files" >
<MarkerIcon>
<FileTextIcon />
</MarkerIcon>
<MarkerContent>Explored 4 files</MarkerContent>
</a>
</Marker>API Reference
Marker
The root marker element. The file also exports markerVariants for composing the marker styles into custom components.
| Prop | Type | Default | Description |
|---|---|---|---|
variant | "default" | "border" | "separator" | "default" | The marker layout. |
render | ReactElement | function | - | Render as a different element, such as a link. |
class | string | - | Additional classes to apply to the root element. |
MarkerIcon
A decorative icon slot. Hidden from assistive tech with aria-hidden.
| Prop | Type | Default | Description |
|---|---|---|---|
class | string | - | Additional classes to apply to the icon slot. |
MarkerContent
The marker text content.
| Prop | Type | Default | Description |
|---|---|---|---|
class | string | - | Additional classes to apply to the content slot. |