All components
Calendar
calendarOrigin UI component.
responsive · 600px
Install
Same command in any shadcn project — React (Vite/CRA), Next.js, Remix, Astro, and more:
$
npx shadcn@latest add https://your-domain/r/calendar.jsonUsage
import { Calendar } from "@/registry/origin-ui/calendar";
export default function Demo() {
return <Calendar />;
}Component source
"use client";
import { ChevronLeftIcon, ChevronRightIcon } from "lucide-react";
import type * as React from "react";
import { DayPicker } from "react-day-picker";
import { cn } from "@/lib/utils";
import { buttonVariants } from "@/components/ui/button";
function Calendar({
className,
classNames,
showOutsideDays = true,
components: userComponents,
...props
}: React.ComponentProps<typeof DayPicker>) {
const defaultClassNames = {
button_next: cn(
buttonVariants({ variant: "ghost" }),
"size-9 p-0 text-muted-foreground/80 hover:text-foreground",
),
button_previous: cn(
buttonVariants({ variant: "ghost" }),
"size-9 p-0 text-muted-foreground/80 hover:text-foreground",
),
caption_label: "text-sm font-medium",
day: "group size-9 px-0 py-px text-sm",
day_button:
"relative flex size-9 items-center justify-center whitespace-nowrap rounded-md p-0 text-foreground group-[[data-selected]:not(.range-middle)]:[transition-property:color,background-color,border-radius,box-shadow] group-[[data-selected]:not(.range-middle)]:duration-150 group-data-disabled:pointer-events-none focus-visible:z-10 hover:not-in-data-selected:bg-accent group-data-selected:bg-primary hover:not-in-data-selected:text-foreground group-data-selected:text-primary-foreground group-data-disabled:text-foreground/30 group-data-disabled:line-through group-data-outside:text-foreground/30 group-data-selected:group-data-outside:text-primary-foreground outline-none focus-visible:ring-ring/50 focus-visible:ring-[3px] group-[.range-start:not(.range-end)]:rounded-e-none group-[.range-end:not(.range-start)]:rounded-s-none group-[.range-middle]:rounded-none group-[.range-middle]:group-data-selected:bg-accent group-[.range-middle]:group-data-selected:text-foreground",
hidden: "invisible",
month: "w-full",
month_caption:
"relative mx-10 mb-1 flex h-9 items-center justify-center z-20",
months: "relative flex flex-col sm:flex-row gap-4",
nav: "absolute top-0 flex w-full justify-between z-10",
outside:
"text-muted-foreground data-selected:bg-accent/50 data-selected:text-muted-foreground",
range_end: "range-end",
range_middle: "range-middle",
range_start: "range-start",
today:
"*:after:pointer-events-none *:after:absolute *:after:bottom-1 *:after:start-1/2 *:after:z-10 *:after:size-[3px] *:after:-translate-x-1/2 *:after:rounded-full *:after:bg-primary [&[data-selected]:not(.range-middle)>*]:after:bg-background [&[data-disabled]>*]:after:bg-foreground/30 *:after:transition-colors",
week_number: "size-9 p-0 text-xs font-medium text-muted-foreground/80",
weekday: "size-9 p-0 text-xs font-medium text-muted-foreground/80",
};
const mergedClassNames: typeof defaultClassNames = Object.keys(
defaultClassNames,
).reduce(
(acc, key) => {
const userClass = classNames?.[key as keyof typeof classNames];
const baseClass =
defaultClassNames[key as keyof typeof defaultClassNames];
acc[key as keyof typeof defaultClassNames] = userClass
? cn(baseClass, userClass)
: baseClass;
return acc;
},
{ ...defaultClassNames } as typeof defaultClassNames,
);
const defaultComponents = {
Chevron: (props: {
className?: string;
size?: number;
disabled?: boolean;
orientation?: "left" | "right" | "up" | "down";
}) => {
if (props.orientation === "left") {
return <ChevronLeftIcon size={16} {...props} aria-hidden="true" />;
}
return <ChevronRightIcon size={16} {...props} aria-hidden="true" />;
},
};
const mergedComponents = {
...defaultComponents,
...userComponents,
};
return (
<DayPicker
className={cn("w-fit", className)}
classNames={mergedClassNames}
components={mergedComponents}
showOutsideDays={showOutsideDays}
{...props}
/>
);
}
export { Calendar };Dependencies
react-day-pickerdate-fns
Source: Origin UI