apps/v4/content/docs/changelog/2025-10-new-components.mdx
For this round of components, I looked at what we build every day, the boring stuff we rebuild over and over, and made reusable abstractions you can actually use.
These components work with every component library, Radix, Base UI, React Aria, you name it. Copy and paste to your projects.
Okay let's start with the easiest ones: Spinner and Kbd. Pretty basic. We all know what they do.
Here's how you render a spinner:
import { Spinner } from "@/components/ui/spinner"
<Spinner />
Here's what it looks like:
<ComponentPreview name="spinner-basic"
/>
Here's what it looks like in a button:
<ComponentPreview name="spinner-button"
/>
You can edit the code and replace it with your own spinner.
<ComponentPreview name="spinner-custom"
/>
Kbd is a component that renders a keyboard key.
import { Kbd, KbdGroup } from "@/components/ui/kbd"
<Kbd>Ctrl</Kbd>
Use KbdGroup to group keyboard keys together.
<KbdGroup>
<Kbd>Ctrl</Kbd>
<Kbd>B</Kbd>
</KbdGroup>
<ComponentPreview name="kbd-demo"
/>
You can add it to buttons, tooltips, input groups, and more.
I got a lot of requests for this one: Button Group. It's a container that groups related buttons together with consistent styling. Great for action groups, split buttons, and more.
<ComponentPreview name="button-group-demo"
/>
Here's the code:
import { ButtonGroup } from "@/components/ui/button-group"
<ButtonGroup>
<Button>Button 1</Button>
<Button>Button 2</Button>
</ButtonGroup>
You can nest button groups to create more complex layouts with spacing.
<ButtonGroup>
<ButtonGroup>
<Button>Button 1</Button>
<Button>Button 2</Button>
</ButtonGroup>
<ButtonGroup>
<Button>Button 3</Button>
<Button>Button 4</Button>
</ButtonGroup>
</ButtonGroup>
Use ButtonGroupSeparator to create split buttons. Classic dropdown pattern.
<ComponentPreview name="button-group-dropdown"
/>
You can also use it to add prefix or suffix buttons and text to inputs.
<ComponentPreview name="button-group-select"
/>
<ButtonGroup>
<ButtonGroupText>Prefix</ButtonGroupText>
<Input placeholder="Type something here..." />
<Button>Button</Button>
</ButtonGroup>
Input Group lets you add icons, buttons, and more to your inputs. You know, all those little bits you always need around your inputs.
import {
InputGroup,
InputGroupAddon,
InputGroupInput,
} from "@/components/ui/input-group"
<InputGroup>
<InputGroupInput placeholder="Search..." />
<InputGroupAddon>
<SearchIcon />
</InputGroupAddon>
</InputGroup>
Here's a preview with icons:
<ComponentPreview name="input-group-icon" className="[&_.preview]:h-[300px] [&_pre]:h-[300px]!" />
You can also add buttons to the input group.
<ComponentPreview name="input-group-button" className="[&_.preview]:h-[300px] [&_pre]:h-[300px]!" />
Or text, labels, tooltips, ...
<ComponentPreview name="input-group-text" className="[&_.preview]:h-[350px] [&_pre]:h-[350px]!" />
It also works with textareas so you can build really complex components with lots of knobs and dials or yet another prompt form.
<ComponentPreview name="input-group-textarea" className="[&_.preview]:h-[450px] [&_pre]:h-[450px]!" />
Oh here are some cool ones with spinners:
<ComponentPreview name="input-group-spinner" className="[&_.preview]:h-[350px] [&_pre]:h-[350px]!" />
Introducing Field, a component for building really complex forms. The abstraction here is beautiful.
It took me a long time to get it right but I made it work with all your form libraries: Server Actions, React Hook Form, TanStack Form, Bring Your Own Form.
import {
Field,
FieldDescription,
FieldError,
FieldLabel,
} from "@/components/ui/field"
Here's a basic field with an input:
<Field>
<FieldLabel htmlFor="username">Username</FieldLabel>
<Input id="username" placeholder="Max Leiter" />
<FieldDescription>
Choose a unique username for your account.
</FieldDescription>
</Field>
<ComponentPreview name="field-input" className="[&_.preview]:h-[350px] [&_pre]:h-[350px]!" />
It works with all form controls. Inputs, textareas, selects, checkboxes, radios, switches, sliders, you name it. Here's a full example:
<ComponentPreview name="field-demo" className="[&_.preview]:h-[850px] [&_pre]:h-[850px]!" />
Here are some checkbox fields:
<ComponentPreview name="field-checkbox" className="[&_.preview]:h-[500px] [&_pre]:h-[500px]!" />
You can group fields together using FieldGroup and FieldSet. Perfect for
multi-section forms.
<FieldSet>
<FieldLegend />
<FieldGroup>
<Field />
<Field />
</FieldGroup>
</FieldSet>
<ComponentPreview name="field-fieldset" className="[&_.preview]:h-[500px] [&_pre]:h-[500px]!" />
Making it responsive is easy. Use orientation="responsive" and it switches
between vertical and horizontal layouts based on container width. Done.
<ComponentPreview name="field-responsive" className="[&_.preview]:h-[600px] [&_pre]:h-[600px]!" />
Wait, here's more. Wrap your fields in FieldLabel to create a selectable field group. Really easy. And it looks great.
<ComponentPreview name="field-choice-card" className="[&_.preview]:h-[600px] [&_pre]:h-[600px]!" />
This one is a straightforward flex container that can house nearly any type of content.
I've built this so many times that I decided to create a component for it. Now I use it all the time. I use it to display lists of items, cards, and more.
import {
Item,
ItemContent,
ItemDescription,
ItemMedia,
ItemTitle,
} from "@/components/ui/item"
Here's a basic item:
<Item>
<ItemMedia variant="icon">
<HomeIcon />
</ItemMedia>
<ItemContent>
<ItemTitle>Dashboard</ItemTitle>
<ItemDescription>Overview of your account and activity.</ItemDescription>
</ItemContent>
</Item>
<ComponentPreview name="item-demo" className="[&_.preview]:h-[300px] [&_.preview]:p-4 [&_pre]:h-[300px]!" />
You can add icons, avatars, or images to the item.
<ComponentPreview name="item-icon" className="[&_.preview]:h-[300px] [&_.preview]:p-4 [&_pre]:h-[300px]!" />
<ComponentPreview name="item-avatar" className="[&_.preview]:h-[300px] [&_.preview]:p-4 [&_pre]:h-[300px]!" />
And here's what a list of items looks like with ItemGroup:
<ComponentPreview name="item-group" className="[&_.preview]:h-[500px] [&_.preview]:p-4 [&_pre]:h-[500px]!" />
Need it as a link? Use the asChild prop:
<Item asChild>
<a href="/dashboard">
<ItemMedia variant="icon">
<HomeIcon />
</ItemMedia>
<ItemContent>
<ItemTitle>Dashboard</ItemTitle>
<ItemDescription>Overview of your account and activity.</ItemDescription>
</ItemContent>
</a>
</Item>
<ComponentPreview name="item-link" className="[&_.preview]:h-[400px] [&_.preview]:p-4 [&_pre]:h-[400px]!" />
Okay last one: Empty. Use this to display empty states in your app.
import {
Empty,
EmptyContent,
EmptyDescription,
EmptyMedia,
EmptyTitle,
} from "@/components/ui/empty"
Here's how you use it:
<Empty>
<EmptyMedia variant="icon">
<InboxIcon />
</EmptyMedia>
<EmptyTitle>No messages</EmptyTitle>
<EmptyDescription>You don't have any messages yet.</EmptyDescription>
<EmptyContent>
<Button>Send a message</Button>
</EmptyContent>
</Empty>
<ComponentPreview name="empty-demo" className="[&_.preview]:h-[400px] [&_.preview]:p-4 [&_pre]:h-[400px]!" />
You can use it with avatars:
<ComponentPreview name="empty-avatar" className="[&_.preview]:h-[400px] [&_pre]:h-[400px]!" />
Or with input groups for things like search results or email subscriptions:
<ComponentPreview name="empty-input-group" className="[&_.preview]:h-[450px] [&_pre]:h-[450px]!" />
That's it. Seven new components. Works with all your libraries. Ready for your projects.