Calendar
Element that enables users to view and interact with dates in a structured and organized format. It allows users to navigate through different months and years, select specific dates, and often provides a visual representation of events or appointments.
Single Calendar
Single Calendar typically allows users to view and interact with dates, navigate through months or years, and select specific date.
July 2024
Su | Mo | Tu | We | Th | Fr | Sa |
---|---|---|---|---|---|---|
30 | 1 | 2 | 3 | 4 | 5 | 6 |
7 | 8 | 9 | 10 | 11 | 12 | 13 |
14 | 15 | 16 | 17 | 18 | 19 | 20 |
21 | 22 | 23 | 24 | 25 | 26 | 27 |
28 | 29 | 30 | 31 | 1 | 2 | 3 |
4 | 5 | 6 | 7 | 8 | 9 | 10 |
"use client";import * as Button from "@lora-ui/button";import * as Calendar from "@lora-ui/calendar";import { twMerge } from "tailwind-merge";export default function Single() {const monthsInYear = ["January","February","March","April","May","June","July","August","September","October","November","December",];const daysInWeekStartWithSunday = {Sun: "Su",Mon: "Mo",Tue: "Tu",Wed: "We",Thu: "Th",Fri: "Fr",Sat: "Sa",};function onValueChange(dates) {console.log(dates);}return (<Calendar.RootdaysInWeek={daysInWeekStartWithSunday}mode="single"defaultDates={[new Date("2024-07-17T00:00:00")]}onValueChange={onValueChange}className="border-box w-[328px] rounded-xl border border-[#eaecf0] bg-[#ffffff] dark:border-[#1f242f] dark:bg-[#0c111d]"><div className="divide-y divide-[#eaecf0] dark:divide-[#1f242f]"><div className="px-6 py-5"><Calendar.Header className="mb-3 flex h-[36px] items-center justify-between">{({ goPreviousMonth, goNextMonth, year, month }) => {return (<><Button.RootonClick={goPreviousMonth()}className="flex items-center justify-center whitespace-nowrap rounded-md p-2 disabled:cursor-not-allowed"aria-label="arrow left"><svgwidth="15"height="15"viewBox="0 0 15 15"fill="none"xmlns="http://www.w3.org/2000/svg"className="text-[#344054] dark:text-[#cecfd2]"><pathd="M8.84182 3.13514C9.04327 3.32401 9.05348 3.64042 8.86462 3.84188L5.43521 7.49991L8.86462 11.1579C9.05348 11.3594 9.04327 11.6758 8.84182 11.8647C8.64036 12.0535 8.32394 12.0433 8.13508 11.8419L4.38508 7.84188C4.20477 7.64955 4.20477 7.35027 4.38508 7.15794L8.13508 3.15794C8.32394 2.95648 8.64036 2.94628 8.84182 3.13514Z"fill="currentColor"fillRule="evenodd"clipRule="evenodd"/></svg></Button.Root><span className="flex items-center whitespace-nowrap text-base font-semibold text-[#344054] dark:text-[#cecfd2]">{monthsInYear[month - 1] + " " + year}</span><Button.RootonClick={goNextMonth()}className="flex items-center justify-center whitespace-nowrap rounded-md p-2 disabled:cursor-not-allowed"aria-label="arrow right"><svgwidth="15"height="15"viewBox="0 0 15 15"fill="none"xmlns="http://www.w3.org/2000/svg"className="text-[#344054] dark:text-[#cecfd2]"><pathd="M6.1584 3.13508C6.35985 2.94621 6.67627 2.95642 6.86514 3.15788L10.6151 7.15788C10.7954 7.3502 10.7954 7.64949 10.6151 7.84182L6.86514 11.8418C6.67627 12.0433 6.35985 12.0535 6.1584 11.8646C5.95694 11.6757 5.94673 11.3593 6.1356 11.1579L9.565 7.49985L6.1356 3.84182C5.94673 3.64036 5.95694 3.32394 6.1584 3.13508Z"fill="currentColor"fillRule="evenodd"clipRule="evenodd"/></svg></Button.Root></>);}}</Calendar.Header><Calendar.Body className="flex w-full flex-col"><Calendar.Days className="flex justify-around">{Object.keys(daysInWeekStartWithSunday).map((day, index) => (<Calendar.Daykey={index}className="flex h-[40px] w-10 items-center justify-center text-sm font-medium text-[#344054] dark:text-[#cecfd2]">{daysInWeekStartWithSunday[day]}</Calendar.Day>))}</Calendar.Days><Calendar.Dates>{({ dates }) => {return dates.map((row, index) => (<tr key={index} className="flex justify-around">{row.map((date, index) => {return (<Calendar.DateCellkey={index}date={date}className="w-10 text-sm font-normal"><divrole="button"tabIndex={date.isDisabled ? -1 : 0}className={twMerge("relative flex h-[40px] items-center justify-center rounded-full",date.isDisabled? "text-[#667085] hover:cursor-not-allowed hover:bg-transparent dark:text-[#85888e]": "hover:[#182230] text-[#344054] hover:cursor-pointer hover:bg-[#f2f4f7] hover:text-sm hover:font-medium dark:text-[#cecfd2] dark:hover:bg-[#1f242f] dark:hover:text-[#ececed]",date.isActived &&"bg-[#7f56d9] text-[#ffffff] hover:bg-[#6941c6] hover:text-sm hover:font-medium hover:text-[#ffffff] dark:bg-[#7f56d9] dark:text-[#f5f5f6] dark:hover:bg-[#9e77ed] dark:hover:text-[#f5f5f6]",)}>{date.date}{date.isMarked && (<divclassName={twMerge("absolute bottom-1 h-[5px] w-[5px] rounded-full",date.isActived? "bg-[#ffffff] dark:bg-[#f5f5f6]": "bg-[#7f56d9] dark:bg-[#7f56d9]",)}/>)}</div></Calendar.DateCell>);})}</tr>));}}</Calendar.Dates></Calendar.Body></div></div></Calendar.Root>);}
Range Calendar
Dual Range Calendar allows users to select dates within a specific range on a calendar that across multiple months
August 2024
Su | Mo | Tu | We | Th | Fr | Sa |
---|---|---|---|---|---|---|
28 | 29 | 30 | 31 | 1 | 2 | 3 |
4 | 5 | 6 | 7 | 8 | 9 | 10 |
11 | 12 | 13 | 14 | 15 | 16 | 17 |
18 | 19 | 20 | 21 | 22 | 23 | 24 |
25 | 26 | 27 | 28 | 29 | 30 | 31 |
1 | 2 | 3 | 4 | 5 | 6 | 7 |
September 2024
Su | Mo | Tu | We | Th | Fr | Sa |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | 5 | 6 | 7 |
8 | 9 | 10 | 11 | 12 | 13 | 14 |
15 | 16 | 17 | 18 | 19 | 20 | 21 |
22 | 23 | 24 | 25 | 26 | 27 | 28 |
29 | 30 | 1 | 2 | 3 | 4 | 5 |
6 | 7 | 8 | 9 | 10 | 11 | 12 |
"use client";import * as Button from "@lora-ui/button";import * as Calendar from "@lora-ui/calendar";import { twMerge } from "tailwind-merge";export default function Range() {const monthsInYear = ["January","February","March","April","May","June","July","August","September","October","November","December",];const daysInWeekStartWithSunday = {Sun: "Su",Mon: "Mo",Tue: "Tu",Wed: "We",Thu: "Th",Fri: "Fr",Sat: "Sa",};function onValueChange(dates) {console.log(dates);}return (<Calendar.RootdaysInWeek={daysInWeekStartWithSunday}onValueChange={onValueChange}defaultDates={[new Date("2024-08-17T00:00:00"),new Date("2024-08-25T00:00:00"),]}className="flex rounded-xl border dark:border-[#1f242f]"mode="range"><div className="border-box w-[328px] px-6 py-5"><Calendar.Header className="mb-3 flex h-[36px] items-center justify-between">{({goPreviousMonth,goNextMonth,fromYear,fromMonth,toYear,toMonth,}) => {return (<><Button.RootonClick={goPreviousMonth("from")}className="flex items-center justify-center whitespace-nowrap rounded-md p-2 disabled:cursor-not-allowed"aria-label="arrow left"><svgwidth="15"height="15"viewBox="0 0 15 15"fill="none"xmlns="http://www.w3.org/2000/svg"className="text-[#344054] dark:text-[#cecfd2]"><pathd="M8.84182 3.13514C9.04327 3.32401 9.05348 3.64042 8.86462 3.84188L5.43521 7.49991L8.86462 11.1579C9.05348 11.3594 9.04327 11.6758 8.84182 11.8647C8.64036 12.0535 8.32394 12.0433 8.13508 11.8419L4.38508 7.84188C4.20477 7.64955 4.20477 7.35027 4.38508 7.15794L8.13508 3.15794C8.32394 2.95648 8.64036 2.94628 8.84182 3.13514Z"fill="currentColor"fillRule="evenodd"clipRule="evenodd"/></svg></Button.Root><span className="flex items-center whitespace-nowrap text-base font-semibold text-[#344054] dark:text-[#cecfd2]">{monthsInYear[fromMonth - 1] + " " + fromYear}</span><Button.RootonClick={goNextMonth("from")}isDisabled={new Date(fromYear, fromMonth + 1, 0).getTime() ===new Date(toYear, toMonth, 0).getTime()}className="group flex items-center justify-center whitespace-nowrap rounded-md p-2 disabled:cursor-not-allowed"aria-label="arrow right"><svgwidth="15"height="15"viewBox="0 0 15 15"fill="none"xmlns="http://www.w3.org/2000/svg"className="text-[#344054] dark:text-[#cecfd2]"><pathd="M6.1584 3.13508C6.35985 2.94621 6.67627 2.95642 6.86514 3.15788L10.6151 7.15788C10.7954 7.3502 10.7954 7.64949 10.6151 7.84182L6.86514 11.8418C6.67627 12.0433 6.35985 12.0535 6.1584 11.8646C5.95694 11.6757 5.94673 11.3593 6.1356 11.1579L9.565 7.49985L6.1356 3.84182C5.94673 3.64036 5.95694 3.32394 6.1584 3.13508Z"fill="currentColor"fillRule="evenodd"clipRule="evenodd"/></svg></Button.Root></>);}}</Calendar.Header><Calendar.Body className="flex w-full flex-col"><Calendar.Days className="flex justify-around">{Object.keys(daysInWeekStartWithSunday).map((day, index) => (<Calendar.Daykey={index}className="flex h-[40px] w-10 items-center justify-center text-sm font-medium text-[#344054] dark:text-[#cecfd2]">{daysInWeekStartWithSunday[day]}</Calendar.Day>))}</Calendar.Days><Calendar.Dates where="from">{({ dates, where }) => {return dates.map((row, index) => (<tr key={index} className="flex justify-around">{row.map((date, index) => (<Calendar.DateCellkey={index}where={where}date={date}className={twMerge("w-10 text-sm font-normal",date.isStart && "rounded-l-full",date.isEnd && "rounded-r-full",date.isSelected &&date.dateObj.getDay() === 0 &&"rounded-l-full",date.isSelected &&date.dateObj.getDay() === 6 &&"rounded-r-full",date.isSelected && "bg-[#f9fafb] dark:bg-[#1f242f]",date.isEntered && "rounded-r-full",)}><divrole="button"tabIndex={date.isDisabled ? -1 : 0}className={twMerge("relative flex h-[40px] items-center justify-center rounded-full",date.isDisabled? "text-[#667085] hover:cursor-not-allowed hover:bg-transparent dark:text-[#85888e]": "text-[#344054] hover:cursor-pointer hover:bg-[#f2f4f7] hover:text-sm hover:font-medium hover:text-[#182230] dark:text-[#cecfd2] dark:hover:bg-[#1f242f] dark:hover:text-[#ececed]",date.isActived &&"bg-[#7f56d9] text-[#ffffff] hover:bg-[#6941c6] hover:text-sm hover:font-medium hover:text-[#ffffff] dark:bg-[#7f56d9] dark:text-[#f5f5f6] dark:hover:bg-[#9e77ed] dark:hover:text-[#f5f5f6]",)}>{date.date}{date.isMarked && (<divclassName={twMerge("absolute bottom-1 h-[5px] w-[5px] rounded-full",date.isActived? "bg-[#ffffff] dark:bg-[#f5f5f6]": "bg-[#7f56d9] dark:bg-[#7f56d9]",)}/>)}</div></Calendar.DateCell>))}</tr>));}}</Calendar.Dates></Calendar.Body></div><div className="border-box w-[328px] px-6 py-5"><Calendar.Header className="mb-3 flex h-[36px] items-center justify-between">{({goPreviousMonth,goNextMonth,fromYear,fromMonth,toYear,toMonth,}) => {return (<><Button.RootonClick={goPreviousMonth("to")}isDisabled={new Date(fromYear, fromMonth + 1, 0).getTime() ===new Date(toYear, toMonth, 0).getTime()}aria-label="arrow left"className="group flex items-center justify-center whitespace-nowrap rounded-md p-2 disabled:cursor-not-allowed"><svgwidth="15"height="15"viewBox="0 0 15 15"fill="none"xmlns="http://www.w3.org/2000/svg"className="text-[#344054] dark:text-[#cecfd2]"><pathd="M8.84182 3.13514C9.04327 3.32401 9.05348 3.64042 8.86462 3.84188L5.43521 7.49991L8.86462 11.1579C9.05348 11.3594 9.04327 11.6758 8.84182 11.8647C8.64036 12.0535 8.32394 12.0433 8.13508 11.8419L4.38508 7.84188C4.20477 7.64955 4.20477 7.35027 4.38508 7.15794L8.13508 3.15794C8.32394 2.95648 8.64036 2.94628 8.84182 3.13514Z"fill="currentColor"fillRule="evenodd"clipRule="evenodd"/></svg></Button.Root><span className="flex items-center whitespace-nowrap text-base font-semibold text-[#344054] dark:text-[#cecfd2]">{monthsInYear[toMonth - 1] + " " + toYear}</span><Button.RootonClick={goNextMonth("to")}aria-label="arrow right"className="flex items-center justify-center whitespace-nowrap rounded-md p-2 disabled:cursor-not-allowed"><svgwidth="15"height="15"viewBox="0 0 15 15"fill="none"xmlns="http://www.w3.org/2000/svg"className="text-[#344054] dark:text-[#cecfd2]"><pathd="M6.1584 3.13508C6.35985 2.94621 6.67627 2.95642 6.86514 3.15788L10.6151 7.15788C10.7954 7.3502 10.7954 7.64949 10.6151 7.84182L6.86514 11.8418C6.67627 12.0433 6.35985 12.0535 6.1584 11.8646C5.95694 11.6757 5.94673 11.3593 6.1356 11.1579L9.565 7.49985L6.1356 3.84182C5.94673 3.64036 5.95694 3.32394 6.1584 3.13508Z"fill="currentColor"fillRule="evenodd"clipRule="evenodd"/></svg></Button.Root></>);}}</Calendar.Header><Calendar.Body className="flex w-full flex-col"><Calendar.Days className="flex justify-around">{Object.keys(daysInWeekStartWithSunday).map((day, index) => (<Calendar.Daykey={index}className="flex h-[40px] w-10 items-center justify-center text-sm font-medium text-[#344054] dark:text-[#cecfd2]">{daysInWeekStartWithSunday[day]}</Calendar.Day>))}</Calendar.Days><Calendar.Dates where="to">{({ dates, where }) => {return dates.map((row, index) => (<tr key={index} className="flex justify-around">{row.map((date, index) => (<Calendar.DateCellkey={index}where={where}date={date}className={twMerge("w-10 text-sm font-normal",date.isStart && "rounded-l-full",date.isEnd && "rounded-r-full",date.isSelected &&date.dateObj.getDay() === 0 &&"rounded-l-full",date.isSelected &&date.dateObj.getDay() === 6 &&"rounded-r-full",date.isSelected && "bg-[#f9fafb] dark:bg-[#1f242f]",date.isEntered && "rounded-r-full",)}><divrole="button"tabIndex={date.isDisabled ? -1 : 0}className={twMerge("relative flex h-[40px] items-center justify-center rounded-full",date.isDisabled? "text-[#667085] hover:cursor-not-allowed hover:bg-transparent dark:text-[#85888e]": "text-[#344054] hover:cursor-pointer hover:bg-[#f2f4f7] hover:text-sm hover:font-medium hover:text-[#182230] dark:text-[#cecfd2] dark:hover:bg-[#1f242f] dark:hover:text-[#ececed]",date.isActived &&"hover: bg-[#7f56d9] text-[#ffffff] hover:bg-[#6941c6] hover:text-[#ffffff] dark:bg-[#7f56d9] dark:text-[#f5f5f6] dark:hover:bg-[#9e77ed] dark:hover:text-[#f5f5f6]",)}>{date.date}{date.isMarked && (<divclassName={twMerge("absolute bottom-1 h-[5px] w-[5px] rounded-full",date.isActived? "bg-[#ffffff] dark:bg-[#f5f5f6]": "bg-[#7f56d9] dark:bg-[#7f56d9]",)}/>)}</div></Calendar.DateCell>))}</tr>));}}</Calendar.Dates></Calendar.Body></div></Calendar.Root>);}
Installation
Install the component from your command line.
npm install @lora-ui/calendar
API Reference
Root
Prop | Type | Default | Explanation |
---|---|---|---|
daysInWeek | Object | { Sun: "Su", Mon: "Mo", Tue: "Tu", Wed: "We", Thu: "Th", Fri: "Fr", Sat: "Sa", } | |
mode | String | "single" | "single" | "range" |
defaultDates | Date[] | Initialed dates by default | |
onValueChange | Function | (dates: Date[]) => {} | Choosed dates |
Header
Prop | Type | Default | Explanation |
---|---|---|---|
children | Function | ({ goPreviousMonth: Function, goNextMonth: Function, year: Number, month: Number }) => {} |
Body
Prop | Type | Default | Explanation |
---|---|---|---|
Days
Prop | Type | Default | Explanation |
---|---|---|---|
Day
Prop | Type | Default | Explanation |
---|---|---|---|
Dates
Prop | Type | Default | Explanation |
---|---|---|---|
children | Function | (dates: Date[]) => {} |
DateCell
Prop | Type | Default | Explanation |
---|---|---|---|
date | Object |
Keyboard
Date Cell
Command | Description |
---|---|
Enter orSpace | Select the date |
ArrowDown | Moves focus to the same day of the next week. |
ArrowUp | Moves focus to the same day of the previous week. |
ArrowRight | Moves focus to the next day. |
ArrowLeft | Moves focus to the previous day. |
Home | Moves focus to the first day (e.g Sunday) of the current week. |
End | Moves focus to the last day (e.g. Saturday) of the current week. |
PageUp |
|
PageDown |
|
Tab | Move focus to the next focusable element |
Shift+Tab | Move focus to the previous focusable element |
Other
All relevant ARIA attributes are automatically managed.