react-datetime-kit
v0.1.0
Published
A modern, accessible React date & time picker library — single date, date range, time, time range, datetime, and datetime range pickers. Works in both JS and TS.
Maintainers
Readme
react-datetime-kit
A modern, accessible, headless-inside / pretty-outside date & time picker toolkit for React. Six picker variants share one API, one theming model, and one keyboard-/screen-reader–friendly core.
🚀 Live Demo
👉 react-datetime-picker.netlify.app
DatePicker · DateRangePicker · TimePicker · TimeRangePicker
DateTimePicker · DateTimeRangePicker- 🎯 TypeScript-first — strict types for every prop and event
- ♿ A11y by default — WAI-ARIA roles, full keyboard nav, focus trap, screen-reader labels
- 🎨 CSS-variable theming — no design-system lock-in, drop your own tokens
- 🌍 i18n + RTL — full
date-fnslocale support, configurable first day of week - ⚡ Tiny, tree-shakeable — uses Floating UI for positioning,
date-fnsas a peer dep - 🧩 Headless inner hooks —
useDatePicker,useDateRange,useTimePickerfor custom UIs - 🌗 Light + dark themes out of the box (auto + manual override)
Table of contents
- Installation
- Quick start
- Pickers
- Common props
- Date-picker props
- Time-picker props
- Theming
- Dark mode
- Localization
- Custom day rendering
- Keyboard shortcuts
- Headless hooks (advanced)
- TypeScript types
- Browser support
- License
Installation
npm install react-datetime-kit
# or
yarn add react-datetime-kit
# or
pnpm add react-datetime-kitPeer dependencies (install if your project does not already have them):
npm install react react-dom date-fns| Dependency | Version |
|------------|---------|
| react | >=17 |
| react-dom | >=17 |
| date-fns | >=2 |
Then import the stylesheet once (typically at your app entry):
import 'react-datetime-kit/styles.css';Quick start
import { useState } from 'react';
import { DatePicker } from 'react-datetime-kit';
import 'react-datetime-kit/styles.css';
export function App() {
const [date, setDate] = useState<Date | null>(new Date());
return (
<DatePicker
value={date}
onChange={setDate}
placeholder="Pick a date"
clearable
/>
);
}That's it. The picker is controlled — you own the state and pass value / onChange. Skip value and pass defaultValue instead for uncontrolled mode.
Pickers
DatePicker
Single calendar-date picker. Popover by default, inline for an always-open calendar.
import { DatePicker } from 'react-datetime-kit';
<DatePicker
value={date}
onChange={setDate}
minDate={new Date('2024-01-01')}
maxDate={new Date('2026-12-31')}
disabledDates={(d) => d.getDay() === 0} // disable Sundays
weekStartsOn={1} // Monday-first calendar
numberOfMonths={2} // two-month view
showWeekNumbers
clearable
placeholder="Select a date"
/>
{/* Inline / always-open variant */}
<DatePicker inline value={date} onChange={setDate} />DateRangePicker
Two-date range with hover preview and optional quick-select sidebar.
import { useState } from 'react';
import { DateRangePicker } from 'react-datetime-kit';
import type { DateRange } from 'react-datetime-kit';
const [range, setRange] = useState<DateRange>({ start: null, end: null });
<DateRangePicker
value={range}
onChange={setRange}
numberOfMonths={2}
showDefaultPresets // Today, Yesterday, Last 7 days, Last 30 days,
// This week, This month, Last month, This year
clearable
/>
{/* Custom presets */}
<DateRangePicker
value={range}
onChange={setRange}
presets={[
{
label: 'Q1 2026',
range: () => ({
start: new Date(2026, 0, 1),
end: new Date(2026, 2, 31),
}),
},
{
label: 'Q2 2026',
range: () => ({
start: new Date(2026, 3, 1),
end: new Date(2026, 5, 30),
}),
},
]}
/>TimePicker
Hour + minute (and optional seconds) column picker.
import { TimePicker } from 'react-datetime-kit';
import type { TimeValue } from 'react-datetime-kit';
const [time, setTime] = useState<TimeValue | null>({ hours: 9, minutes: 30 });
<TimePicker
value={time}
onChange={setTime}
hourFormat={12} // 12 or 24 (default: 24)
minuteStep={5} // 1 | 5 | 15 | 30 …
showSeconds // adds seconds column
clearable
/>TimeRangePicker
Two TimePickers side-by-side for a start/end window.
import { TimeRangePicker } from 'react-datetime-kit';
import type { TimeRange } from 'react-datetime-kit';
const [range, setRange] = useState<TimeRange>({
start: { hours: 9, minutes: 0 },
end: { hours: 17, minutes: 0 },
});
<TimeRangePicker
value={range}
onChange={setRange}
hourFormat={24}
minuteStep={15}
clearable
/>DateTimePicker
Calendar + time columns in one popover. Returns a single Date (date and time merged).
import { DateTimePicker } from 'react-datetime-kit';
<DateTimePicker
value={dt}
onChange={setDt}
hourFormat={12}
minuteStep={5}
showSeconds={false}
clearable
/>DateTimeRangePicker
Range picker with separate start/end time editors. Returns { start, end } of Date.
import { DateTimeRangePicker } from 'react-datetime-kit';
import type { DateRange } from 'react-datetime-kit';
const [range, setRange] = useState<DateRange>({ start: null, end: null });
<DateTimeRangePicker
value={range}
onChange={setRange}
numberOfMonths={2}
hourFormat={24}
minuteStep={15}
showDefaultPresets
clearable
/>Common props
These props are shared by every picker.
| Prop | Type | Default | Description |
|------|------|---------|-------------|
| inline | boolean | false | Render an always-open calendar instead of an input + popover. |
| disabled | boolean | false | Disable the whole picker. |
| readOnly | boolean | false | Show the value but prevent edits. |
| clearable | boolean | false | Show an × button on the input when a value exists. |
| placeholder | string | (varies) | Input placeholder when there is no value. |
| size | 'sm' \| 'md' \| 'lg' | 'md' | Input size variant. |
| theme | 'light' \| 'dark' \| 'auto' | 'auto' | Color scheme override. 'auto' follows OS / data attribute. |
| dir | 'ltr' \| 'rtl' | 'ltr' | Text direction. |
| locale | Locale | (en-US-ish) | A date-fns Locale object. |
| className | string | — | Extra className on the root wrapper. |
| inputClassName | string | — | Extra className on the <input> element. |
| popoverClassName | string | — | Extra className on the floating panel. |
| id | string | — | id attribute for the input. |
| name | string | — | name attribute (useful inside <form>). |
| autoFocus | boolean | false | Focus the input on mount. |
Date-picker props
Apply to DatePicker, DateRangePicker, DateTimePicker, DateTimeRangePicker.
| Prop | Type | Default | Description |
|------|------|---------|-------------|
| value | Date \| null or DateRange | — | Controlled value (range variants use { start, end }). |
| defaultValue | same as value | — | Uncontrolled initial value. |
| onChange | (v) => void | — | Fires on selection change. |
| minDate | Date | — | Earliest selectable date (inclusive). |
| maxDate | Date | — | Latest selectable date (inclusive). |
| disabledDates | Date[] \| (d: Date) => boolean | — | Disable individual dates or by predicate. |
| format | string | locale default | A date-fns format string for the input display. |
| weekStartsOn | 0..6 | 0 (Sun) | First day of the week. |
| showWeekNumbers | boolean | false | Show ISO-8601 week numbers in the calendar. |
| numberOfMonths | number | 1 (range: 2) | Months visible side-by-side. |
| renderDay | (d, defaultNode) => ReactNode | — | Custom render for a day cell — see Custom day rendering. |
Range variants additionally accept:
| Prop | Type | Description |
|------|------|-------------|
| presets | Preset[] | Custom quick-select sidebar entries. |
| showDefaultPresets | boolean | Show built-in presets (Today, Last 7 days, This month, etc.). |
// Preset shape
interface Preset {
label: string;
range: () => { start: Date | null; end: Date | null };
}Time-picker props
Apply to TimePicker, TimeRangePicker, DateTimePicker, DateTimeRangePicker.
| Prop | Type | Default | Description |
|------|------|---------|-------------|
| hourFormat | 12 \| 24 | 24 | 12-hour clock with AM/PM toggle, or 24-hour clock. |
| minuteStep | number | 1 | Minute increments (e.g. 5, 15, 30). |
| secondStep | number | 1 | Second increments (only used when showSeconds). |
| showSeconds | boolean | false | Show the seconds column. |
| format | string | auto-derived | Custom date-fns format string for the input display. |
Theming
All visual styles are driven by CSS variables. Override them on :root (global) or on any ancestor element (scoped). No JS theme config, no Tailwind required in the consumer's project.
:root {
/* Colors */
--rdk-color-primary: #7c3aed;
--rdk-color-primary-hover: #6d28d9;
--rdk-color-primary-soft: #f3e8ff;
--rdk-color-bg: #fafafa;
--rdk-color-surface: #ffffff;
--rdk-color-surface-hover: #f5f3ff;
--rdk-color-text: #18181b;
--rdk-color-text-muted: #71717a;
--rdk-color-border: #e4e4e7;
--rdk-color-border-strong: #d4d4d8;
--rdk-color-danger: #ef4444;
--rdk-color-disabled: #d4d4d8;
--rdk-color-range-bg: #f3e8ff;
/* Radius */
--rdk-radius: 0.75rem;
--rdk-radius-sm: 0.5rem;
--rdk-radius-lg: 1.25rem;
/* Typography */
--rdk-font: 'Inter', system-ui, sans-serif;
/* Shadows */
--rdk-shadow: 0 1px 3px rgba(24, 24, 27, 0.06), 0 4px 12px rgba(124, 58, 237, 0.06);
--rdk-shadow-lg: 0 24px 56px -16px rgba(124, 58, 237, 0.22),
0 12px 24px -8px rgba(24, 24, 27, 0.08);
}Scoped theming
Wrap a section of your app to give it its own picker theme without affecting the rest:
<div data-rdk-theme="dark">
<DatePicker value={date} onChange={setDate} />
</div>Dark mode
Three opt-in strategies — pick whichever matches your stack.
1. Follow the OS color scheme (default — uses prefers-color-scheme)
Nothing to do. Set theme="auto" (the default) and the picker follows @media (prefers-color-scheme: dark).
2. Manual switch via a data attribute (works with most theme switchers)
useEffect(() => {
document.documentElement.setAttribute(
'data-rdk-theme',
isDark ? 'dark' : 'light',
);
}, [isDark]);3. Per-picker override
<DatePicker theme="dark" value={date} onChange={setDate} />Localization
Pass any date-fns locale via the locale prop. Combine with weekStartsOn for full regional accuracy.
import { fr } from 'date-fns/locale';
<DatePicker
value={date}
onChange={setDate}
locale={fr}
weekStartsOn={1} // Monday-first
format="dd MMMM yyyy" // → "25 mai 2026"
/>RTL
<DatePicker dir="rtl" value={date} onChange={setDate} />The icon, navigation arrows, and range-edge rounding all flip automatically.
Custom day rendering
Need to highlight holidays, show event dots, or completely take over the cell? Use renderDay. You receive the date plus the default node so you can decorate without re-implementing.
const holidays = ['2026-01-01', '2026-07-04', '2026-12-25'];
<DatePicker
value={date}
onChange={setDate}
renderDay={(d, defaultNode) => {
const iso = d.toISOString().slice(0, 10);
const isHoliday = holidays.includes(iso);
return (
<>
{defaultNode}
{isHoliday && (
<span
style={{
position: 'absolute',
bottom: 2,
left: '50%',
transform: 'translateX(-50%)',
width: 4,
height: 4,
borderRadius: '50%',
background: '#ef4444',
}}
/>
)}
</>
);
}}
/>Keyboard shortcuts
When the calendar is focused:
| Key | Action |
|-----|--------|
| ← → | Move focus by one day |
| ↑ ↓ | Move focus by one week |
| Home / End | Jump to the start / end of the week |
| PageUp / PageDown | Previous / next month |
| Shift + PageUp/Down | Previous / next year |
| Enter / Space | Select the focused day |
| Escape | Close the popover |
| Tab | Move focus out of the calendar |
Time columns support ↑ / ↓ (or scroll) to increment / decrement.
Headless hooks (advanced)
Need a fully custom UI but want the date math, controllability, and disabled-date logic for free? Drop down to the hook layer.
import { useDatePicker } from 'react-datetime-kit';
function MyCustomPicker() {
const ctrl = useDatePicker({
value: date,
onChange: setDate,
minDate: new Date(),
});
return (
<div>
<button onClick={() => ctrl.setVisibleMonth(addMonths(ctrl.visibleMonth, -1))}>
←
</button>
<span>{format(ctrl.visibleMonth, 'MMMM yyyy')}</span>
<button onClick={() => ctrl.setVisibleMonth(addMonths(ctrl.visibleMonth, 1))}>
→
</button>
{/* Render your own grid using ctrl.isDisabled(d), ctrl.setValue(d), etc. */}
</div>
);
}Available headless hooks:
useDatePicker({ value, onChange, minDate, maxDate, disabledDates })useDateRange({ value, onChange, minDate, maxDate, disabledDates })useTimePicker({ value, onChange })useCalendarKeyboard({ onMove, onSelect, onCancel })
TypeScript types
Everything is fully typed. The most commonly imported types:
import type {
DatePickerProps,
DateRangePickerProps,
TimePickerProps,
TimeRangePickerProps,
DateTimePickerProps,
DateTimeRangePickerProps,
DateRange, // { start: Date | null; end: Date | null }
TimeValue, // { hours: number; minutes: number; seconds?: number }
TimeRange, // { start: TimeValue | null; end: TimeValue | null }
Preset, // { label: string; range: () => DateRange }
Theme, // 'light' | 'dark' | 'auto'
Size, // 'sm' | 'md' | 'lg'
WeekStartsOn, // 0 | 1 | 2 | 3 | 4 | 5 | 6
Locale, // re-exported from date-fns
} from 'react-datetime-kit';Browser support
Modern evergreen browsers (Chrome, Edge, Firefox, Safari — last two stable versions). Requires CSS color-mix() support for opacity-modifier utilities — Chrome 111+, Firefox 113+, Safari 16.4+.
License
MIT © react-datetime-kit MIT License - Copyright (c) 2026 Yogesh Gabani
Built by Yogesh Gabani.
