Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(calendar): add custom cell content to Calendar #3554

Open
wants to merge 31 commits into
base: canary
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 24 commits
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
a24b443
feat(calendar): add custom cell content
1amageek Jun 24, 2024
487be5e
refactor(calendar-cell): add useRef hook to improve performance
1amageek Jun 24, 2024
7a0fa5c
refactor: improve CalendarCell by destructuring props and adding type…
1amageek Jun 25, 2024
a449568
test(calendar): add test for custom cell content in Calendar and Rang…
1amageek Jun 25, 2024
0218de1
chore: add changeset for custom cell content feature in Calendar and …
1amageek Jun 25, 2024
4d40ddd
refactor: use shared template component for custom cell content in ca…
1amageek Jun 29, 2024
c0f2b34
feat: add calendar width and custom cell templates for calendar and r…
1amageek Jul 25, 2024
448ab55
feat(calendar): add renderCellContent prop for custom cell content
1amageek Jul 25, 2024
9445d2b
feat(calendar): move renderCellContent to context
1amageek Nov 22, 2024
930d4af
feat(calendar): add cell components and context management
1amageek Nov 23, 2024
9fc7840
feat(calendar): implement custom cell content rendering
1amageek Nov 23, 2024
b327e40
feat(docs): add custom cell content examples for calendar components
1amageek Nov 23, 2024
b2fcf12
fix(theme): correct typo in calendar cellBody height class
1amageek Nov 23, 2024
39dd82c
Merge branch 'canary' into canary
1amageek Nov 23, 2024
0f22c26
feat(calendar): add day of week calculation to CalendarCell component
1amageek Nov 23, 2024
2ae85a9
refactor(calendar): improve calendar cell components and remove unuse…
1amageek Nov 23, 2024
01b3ba2
feat(calendar): update components to use CalendarCellHeader
1amageek Nov 23, 2024
874af0b
refactor(theme): update gridBodyRow styles for better spacing
1amageek Nov 23, 2024
e814634
refactor(calendar): integrate CalendarCellHeader into calendar cell c…
1amageek Nov 23, 2024
24ccded
feat(calendar): add support for custom cell content in calendar compo…
1amageek Nov 23, 2024
1cd1d21
feat(calendar): add role and aria-label to birthday events
1amageek Nov 27, 2024
8902936
feat(calendar): add calendar cell components and context support
1amageek Dec 7, 2024
84a85d7
feat(calendar): add calendar cell components and context support
1amageek Dec 7, 2024
9e6ca20
feat(calendar): update documentation
1amageek Jan 2, 2025
9091f51
chore: update changelog for calendar customization
1amageek Jan 16, 2025
b920894
feat(calendar): add calendar cell
1amageek Jan 16, 2025
a16dcf4
fix(calendar): fix conflicts
1amageek Jan 20, 2025
5b46ae7
fix(calendar): fix conflicts
1amageek Jan 20, 2025
6a20973
Merge branch 'canary' into canary
1amageek Jan 20, 2025
b62c31c
fix(calendar): calendar-cell-header
1amageek Jan 20, 2025
0d443a0
fix(calendar): heroui
1amageek Jan 20, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .changeset/popular-seals-appear.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@nextui-org/calendar": minor
---

Added the `renderCellContent` prop to the `Calendar` and `RangeCalendar` components, allowing developers to specify custom content for each calendar cell. Updated the existing tests and added new test cases to cover the custom cell content functionality.
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
import React from "react";
import {
Calendar,
CalendarCellContent,
CalendarCellHeader,
CalendarCellBody,
} from "@nextui-org/react";

export default function App() {
return (
<Calendar calendarWidth={400}>
{(date) => (
<CalendarCellContent>
<CalendarCellHeader />
<CalendarCellBody>
<div className="flex flex-col w-full text-tiny gap-0.5 px-0.5">
{date.day % 7 === 0 && (
<span
aria-label="Calendar event"
className="bg-red-500/20 w-full rounded-md px-1 text-red-400 line-clamp-1"
role="status"
>
MTG
</span>
)}
{date.day % 5 === 0 && (
<span
aria-label="calendar event"
className="bg-green-500/20 w-full rounded-md px-1 text-green-400 line-clamp-1"
role="status"
>
MTG
</span>
)}
{date.day % 3 === 0 && (
<span
aria-label="calendar event"
className="bg-yellow-500/20 w-full rounded-md px-1 text-yellow-400 line-clamp-1"
role="status"
>
MTG
</span>
)}
1amageek marked this conversation as resolved.
Show resolved Hide resolved
</div>
</CalendarCellBody>
</CalendarCellContent>
)}
</Calendar>
);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import App from "./custom-cell-content.raw.jsx?raw";

const react = {
"/App.jsx": App,
};

export default {
...react,
};
2 changes: 2 additions & 0 deletions apps/docs/content/components/calendar/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import internationalCalendars from "./international-calendars";
import visibleMonths from "./visible-months";
import pageBehaviour from "./page-behaviour";
import presets from "./presets";
import customCellContent from "./custom-cell-content";

export const calendarContent = {
usage,
Expand All @@ -28,4 +29,5 @@ export const calendarContent = {
visibleMonths,
pageBehaviour,
presets,
customCellContent,
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import {
RangeCalendar,
CalendarCellContent,
CalendarCellHeader,
CalendarCellBody,
} from "@nextui-org/react";

export default function App() {
return (
<RangeCalendar calendarWidth={400}>
{(date) => (
<CalendarCellContent>
<CalendarCellHeader />
<CalendarCellBody>
<div className="flex flex-col w-full text-tiny gap-0.5 px-0.5">
{date.day % 7 === 0 && (
<span
aria-label="calendar event"
className="bg-red-500/20 w-full rounded-md px-1 text-red-400 line-clamp-1"
role="status"
>
MTG
</span>
)}
{date.day % 5 === 0 && (
<span
aria-label="calendar event"
className="bg-green-500/20 w-full rounded-md px-1 text-green-400 line-clamp-1"
role="status"
>
MTG
</span>
)}
{date.day % 3 === 0 && (
<span
aria-label="calendar event"
className="bg-yellow-500/20 w-full rounded-md px-1 text-yellow-400 line-clamp-1"
role="status"
>
MTG
</span>
)}
</div>
</CalendarCellBody>
</CalendarCellContent>
)}
</RangeCalendar>
);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import App from "./custom-cell-content.raw.jsx?raw";

const react = {
"/App.jsx": App,
};

export default {
...react,
};
2 changes: 2 additions & 0 deletions apps/docs/content/components/range-calendar/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import internationalCalendars from "./international-calendars";
import visibleMonths from "./visible-months";
import pageBehaviour from "./page-behaviour";
import presets from "./presets";
import customCellContent from "./custom-cell-content";
import withMonthAndYearPicker from "./with-month-and-year-picker";

export const rangeCalendarContent = {
Expand All @@ -29,5 +30,6 @@ export const rangeCalendarContent = {
visibleMonths,
pageBehaviour,
presets,
customCellContent,
withMonthAndYearPicker,
};
18 changes: 18 additions & 0 deletions apps/docs/content/docs/components/calendar.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,20 @@ Here's the example to customize `topContent` and `bottomContent` to have some pr

<CodeDemo title="Presets" files={calendarContent.presets} />

### Custom Cell Content

The Calendar component supports customizing the cell content in two ways:

<CodeDemo title="Custom Cell" files={calendarContent.customCellContent} />

1amageek marked this conversation as resolved.
Show resolved Hide resolved
The Calendar provides three components for cell customization:

- `CalendarCellContent`: The wrapper component for the cell content
- `CalendarCellHeader`: The interactive header element that handles selection
- `CalendarCellBody`: Additional content container below the button

These components inherit all calendar states (selected, disabled, etc.) and maintain proper accessibility.

## Slots

- **base**: Calendar wrapper, it handles alignment, placement, and general appearance.
Expand All @@ -146,6 +160,10 @@ Here's the example to customize `topContent` and `bottomContent` to have some pr
- **gridBodyRow**: The date grid body row element (e.g. `<tr>`).
- **cell**: The date grid cell element (e.g. `<td>`).
- **cellButton**: The button element within the cell.
- **cellContent**: The wrapper for custom cell content.
- **cellHeaderWrapper**: The wrapper for the cell header content.
- **cellHeader**: The header element within the cell that handles selection.
- **cellBody**: The container for additional cell content.
1amageek marked this conversation as resolved.
Show resolved Hide resolved
- **pickerWrapper**: The wrapper for the picker
- **pickerMonthList**: The month list picker.
- **pickerYearList**: The year list picker.
Expand Down
74 changes: 72 additions & 2 deletions apps/docs/content/docs/components/range-calendar.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ A range calendar consists of a grouping element containing one or more date grid

---


<CarbonAd/>

## Installation
Expand All @@ -32,8 +33,8 @@ A range calendar consists of a grouping element containing one or more date grid

<ImportTabs
commands={{
main: 'import {RangeCalendar} from "@nextui-org/react";',
individual: 'import {RangeCalendar} from "@nextui-org/calendar";',
main: 'import {RangeCalendar, CalendarCellContent, CalendarCellHeader, CalendarCellBody} from "@nextui-org/react";',
individual: 'import {RangeCalendar, CalendarCellContent, CalendarCellHeader, CalendarCellBody} from "@nextui-org/calendar";',
}}
/>

Expand All @@ -45,6 +46,56 @@ Date values are provided using objects in the [@internationalized/date](https://

<CodeDemo title="Usage" files={rangeCalendarContent.usage} />

### Custom Cell Content

The Calendar component supports customizing the cell content in two ways:

1. Using a render function:
```tsx
<RangeCalendar>
{(date, cellState) => (
<CalendarCellContent>
<CalendarCellHeader>
<span
className={
getDayOfWeek(date, locale) === 0
? "text-red-500"
: "text-default-500"
}
>
{date.day}
</span>
</CalendarCellHeader>
</CalendarCellContent>
)}
</RangeCalendar>
```

2. Using component composition:
```tsx
<RangeCalendar>
<CalendarCellContent>
<CalendarCellHeader />
<CalendarCellBody>
<div className="flex flex-col w-full gap-0.5 justify-center items-center p-0.5">
<span className="bg-red-600 h-1 w-full rounded-full" />
<span className="bg-green-600 h-1 w-full rounded-full" />
<span className="bg-yellow-600 h-1 w-full rounded-full" />
</div>
</CalendarCellBody>
</CalendarCellContent>
</RangeCalendar>
```

The calendar provides three components for cell customization:
- `CalendarCellContent`: Wrapper component for the entire cell content
- `CalendarCellHeader`: Interactive button element that handles selection
- `CalendarCellBody`: Container for additional content below the button

These components inherit all calendar states (selected, disabled, etc.) and maintain proper accessibility.

<CodeDemo title="Custom Cell Content" files={rangeCalendarContent.customCellContent} />

### Disabled

The `isDisabled` boolean prop makes the Calendar disabled. Cells cannot be focused or selected.
Expand Down Expand Up @@ -137,6 +188,21 @@ Here's the example to customize `topContent` and `bottomContent` to have some pr

<CodeDemo title="Presets" files={rangeCalendarContent.presets} />

### Custom Cell Content

The Calendar component supports customizing the cell content in two ways:

<CodeDemo title="Custom Cell" files={rangeCalendarContent.customCellContent} />

The Calendar provides three components for cell customization:

- `CalendarCellContent`: The wrapper component for the cell content
- `CalendarCellHeader`: The interactive header element that handles selection
- `CalendarCellBody`: Additional content container below the button

These components inherit all calendar states (selected, disabled, etc.) and maintain proper accessibility.

1amageek marked this conversation as resolved.
Show resolved Hide resolved

## Slots

- **base**: Calendar wrapper, it handles alignment, placement, and general appearance.
Expand All @@ -154,6 +220,10 @@ Here's the example to customize `topContent` and `bottomContent` to have some pr
- **gridBodyRow**: The date grid body row element (e.g. `<tr>`).
- **cell**: The date grid cell element (e.g. `<td>`).
- **cellButton**: The button element within the cell.
- **cellContent**: The wrapper for custom cell content.
- **cellHeaderWrapper**: The wrapper for the cell header content.
- **cellHeader**: The header element within the cell that handles selection.
- **cellBody**: The container for additional cell content.
1amageek marked this conversation as resolved.
Show resolved Hide resolved
- **pickerWrapper**: The wrapper for the picker
- **pickerMonthList**: The month list picker.
- **pickerYearList**: The year list picker.
Expand Down
21 changes: 21 additions & 0 deletions packages/components/calendar/__tests__/calendar.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -464,4 +464,25 @@ describe("Calendar", () => {
expect(year).toHaveAttribute("data-value", "2567");
});
});

describe("Custom cell content", () => {
1amageek marked this conversation as resolved.
Show resolved Hide resolved
it("should render custom content in the calendar cells", () => {
const wrapper = render(
<Calendar defaultValue={new CalendarDate(2024, 3, 31)}>
{(date) => (
<div>
{date.day}
<span>*</span>
</div>
)}
</Calendar>,
);

const gridCells = wrapper.getAllByRole("gridcell");
const customContentCell = gridCells.find((cell) => cell.textContent === "31*");

expect(customContentCell).not.toBeNull();
expect(customContentCell).toHaveTextContent("31*");
});
});
});
26 changes: 26 additions & 0 deletions packages/components/calendar/__tests__/range-calendar.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -748,4 +748,30 @@ describe("RangeCalendar", () => {
expect(end).toEqual(new CalendarDate(2019, 6, 25));
});
});

describe("Custom cell content", () => {
it("should render custom content in the range calendar cells", () => {
const wrapper = render(
<RangeCalendar
defaultValue={{start: new CalendarDate(2024, 6, 25), end: new CalendarDate(2024, 6, 26)}}
>
{(date) => (
<div>
{date.day}
<span>*</span>
</div>
)}
</RangeCalendar>,
);

const gridCells = wrapper.getAllByRole("gridcell");
const customContentCellA = gridCells.find((cell) => cell.textContent === "25*");
const customContentCellB = gridCells.find((cell) => cell.textContent === "26*");

expect(customContentCellA).not.toBeNull();
expect(customContentCellA).toHaveTextContent("25*");
expect(customContentCellB).not.toBeNull();
expect(customContentCellB).toHaveTextContent("26*");
});
});
});
29 changes: 29 additions & 0 deletions packages/components/calendar/src/calendar-cell-body.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
import type {HTMLNextUIProps} from "@nextui-org/system";

import React from "react";

import {useCalendarContext} from "./calendar-context";

interface Props extends HTMLNextUIProps<"div"> {
children: React.ReactNode;
}

export type CalendarCellBodyProps = Props;

export const CalendarCellBody = React.forwardRef<HTMLDivElement, CalendarCellBodyProps>(
({children, ...props}, ref) => {
const {slots, classNames} = useCalendarContext();
const bodyProps = {
...props,
ref: ref,
className: slots?.cellBody({class: classNames?.cellBody}),
"data-slot": "cell-body",
};

return <div {...bodyProps}>{children}</div>;
},
);

CalendarCellBody.displayName = "NextUI.CalendarCellBody";

export default CalendarCellBody;
Loading
Loading