Components
Popover
Displays rich content in a portal, triggered by a button.
Loading...
Installation
CLI
npx shadcn@latest add https://exawizards.com/exabase/design/registry/popover.jsonManual
Copy and paste the following code into your project.
"use client"
import * as React from "react"
import { Popover as PopoverPrimitive } from "@base-ui/react/popover"
import { cn } from "@/lib/utils"
function Popover({ ...props }: PopoverPrimitive.Root.Props) {
return <PopoverPrimitive.Root data-slot="popover" {...props} />
}
function PopoverTrigger({ ...props }: PopoverPrimitive.Trigger.Props) {
return <PopoverPrimitive.Trigger data-slot="popover-trigger" {...props} />
}
function PopoverContent({
children,
className,
align = "center",
alignOffset = 0,
side = "bottom",
sideOffset = 4,
collisionPadding = 4,
arrow,
...props
}: PopoverPrimitive.Popup.Props &
Pick<
PopoverPrimitive.Positioner.Props,
"align" | "alignOffset" | "side" | "sideOffset" | "collisionPadding"
> & {
arrow?: boolean
}) {
return (
<PopoverPrimitive.Portal>
<PopoverPrimitive.Positioner
align={align}
alignOffset={alignOffset}
side={side}
sideOffset={sideOffset}
className="isolate z-50"
>
<PopoverPrimitive.Popup
data-slot="popover-content"
className={cn(
"z-50 flex w-72 origin-(--transform-origin) flex-col gap-2.5 rounded-lg bg-popover p-2.5 text-sm text-popover-foreground shadow-md outline outline-border duration-100 data-[side=bottom]:slide-in-from-top-2 data-[side=inline-end]:slide-in-from-left-2 data-[side=inline-start]:slide-in-from-right-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 data-open:animate-in data-open:fade-in-0 data-open:zoom-in-95 data-closed:animate-out data-closed:fade-out-0 data-closed:zoom-out-95",
className
)}
{...props}
>
{children}
{arrow && (
<PopoverPrimitive.Arrow
data-slot="popover-arrow"
className="data-[side=bottom]:top-[-8px] data-[side=bottom]:rotate-180 data-[side=left]:right-[-13px] data-[side=left]:rotate-90 data-[side=right]:left-[-13px] data-[side=right]:-rotate-90 data-[side=top]:bottom-[-8px]"
>
<svg
width="1rem"
height="0.5rem"
viewBox="0 0 16 8"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M14.7929 0.5L8 7.29289L1.20711 0.5L14.7929"
className="fill-popover stroke-border"
/>
<path
d="M2.41422 1L1.41422 0H14.5858L13.5858 1H2.41422Z"
className="fill-popover"
/>
</svg>
</PopoverPrimitive.Arrow>
)}
</PopoverPrimitive.Popup>
</PopoverPrimitive.Positioner>
</PopoverPrimitive.Portal>
)
}
function PopoverHeader({ className, ...props }: React.ComponentProps<"div">) {
return (
<div
data-slot="popover-header"
className={cn("flex flex-col gap-0.5 text-sm", className)}
{...props}
/>
)
}
function PopoverTitle({ className, ...props }: PopoverPrimitive.Title.Props) {
return (
<PopoverPrimitive.Title
data-slot="popover-title"
className={cn("font-medium", className)}
{...props}
/>
)
}
function PopoverDescription({
className,
...props
}: PopoverPrimitive.Description.Props) {
return (
<PopoverPrimitive.Description
data-slot="popover-description"
className={cn("text-muted-foreground", className)}
{...props}
/>
)
}
export {
Popover,
PopoverContent,
PopoverDescription,
PopoverHeader,
PopoverTitle,
PopoverTrigger,
}
Usage
import {
Popover,
PopoverContent,
PopoverTrigger,
} from "@/components/ui/popover"<Popover>
<PopoverTrigger>Open</PopoverTrigger>
<PopoverContent>Place content for the popover here.</PopoverContent>
</Popover>Examples
Align
Use the align prop on PopoverContent to control the horizontal alignment.
Loading...
With Arrow
Use the arrow prop on PopoverContent to add an arrow.
Loading...
With Form
A popover with form fields inside.
Loading...
API
See the Base UI documentation for more information.
The PopoverContent component has additional property:
| Prop | Type | Defalt | Description |
|---|---|---|---|
arrow | boolean | false | If true, adds an arrow to the menu. |