Date Picker
A component for selecting a date from a calendar interface.
Anatomy
Usage
<script>
  import {DatePicker} from 'ui-ingredients';
  import {CalendarIcon, ChevronLeftIcon, ChevronRightIcon} from '$lib/icons';
  /** @type {import('ui-ingredients').DateValue[]} */
  let value = $state([DatePicker.parse('1990-01-01')]);
</script>
<DatePicker.Root
  {value}
  onValueChange={(detail) => {
    value = detail.value;
  }}
>
  {#snippet children(context)}
    <DatePicker.Label>Label</DatePicker.Label>
    <DatePicker.Control>
      <DatePicker.Input />
      <DatePicker.Trigger>
        <CalendarIcon />
      </DatePicker.Trigger>
    </DatePicker.Control>
    <DatePicker.Positioner>
      <DatePicker.Content>
        <DatePicker.View view="day">
          <DatePicker.ViewControl>
            <DatePicker.PrevTrigger>
              <ChevronLeftIcon />
            </DatePicker.PrevTrigger>
            <DatePicker.ViewTrigger>
              <DatePicker.RangeText />
            </DatePicker.ViewTrigger>
            <DatePicker.NextTrigger>
              <ChevronRightIcon />
            </DatePicker.NextTrigger>
          </DatePicker.ViewControl>
          <DatePicker.Table>
            <DatePicker.TableHead>
              <DatePicker.TableRow>
                {#each context.weekDays as weekDay}
                  <DatePicker.TableHeader>
                    {weekDay.narrow}
                  </DatePicker.TableHeader>
                {/each}
              </DatePicker.TableRow>
            </DatePicker.TableHead>
            <DatePicker.TableBody>
              {#each context.weeks as week}
                <DatePicker.TableRow>
                  {#each week as day}
                    <DatePicker.DayTableCell value={day}>
                      <DatePicker.DayTableCellTrigger>
                        {day.day}
                      </DatePicker.DayTableCellTrigger>
                    </DatePicker.DayTableCell>
                  {/each}
                </DatePicker.TableRow>
              {/each}
            </DatePicker.TableBody>
          </DatePicker.Table>
        </DatePicker.View>
        <!-- MONTH -->
        <DatePicker.View view="month">
          <DatePicker.ViewControl>
            <DatePicker.PrevTrigger>
              <ChevronLeftIcon />
            </DatePicker.PrevTrigger>
            <DatePicker.ViewTrigger>
              <DatePicker.RangeText />
            </DatePicker.ViewTrigger>
            <DatePicker.NextTrigger>
              <ChevronRightIcon />
            </DatePicker.NextTrigger>
          </DatePicker.ViewControl>
          <DatePicker.Table>
            <DatePicker.TableBody>
              {#each context.getMonthsGrid( {columns: 4, format: 'short'}, ) as months}
                <DatePicker.TableRow>
                  {#each months as month}
                    <DatePicker.MonthTableCell value={month.value}>
                      <DatePicker.MonthTableCellTrigger>
                        {month.label}
                      </DatePicker.MonthTableCellTrigger>
                    </DatePicker.MonthTableCell>
                  {/each}
                </DatePicker.TableRow>
              {/each}
            </DatePicker.TableBody>
          </DatePicker.Table>
        </DatePicker.View>
        <!-- YEAR -->
        <DatePicker.View view="year">
          <DatePicker.ViewControl>
            <DatePicker.PrevTrigger>
              <ChevronLeftIcon />
            </DatePicker.PrevTrigger>
            <DatePicker.ViewTrigger>
              <DatePicker.RangeText />
            </DatePicker.ViewTrigger>
            <DatePicker.NextTrigger>
              <ChevronRightIcon />
            </DatePicker.NextTrigger>
          </DatePicker.ViewControl>
          <DatePicker.Table>
            <DatePicker.TableBody>
              {#each context.getYearsGrid({columns: 4}) as years}
                <DatePicker.TableRow>
                  {#each years as year}
                    <DatePicker.YearTableCell value={year.value}>
                      <DatePicker.YearTableCellTrigger>
                        {year.label}
                      </DatePicker.YearTableCellTrigger>
                    </DatePicker.YearTableCell>
                  {/each}
                </DatePicker.TableRow>
              {/each}
            </DatePicker.TableBody>
          </DatePicker.Table>
        </DatePicker.View>
      </DatePicker.Content>
    </DatePicker.Positioner>
  {/snippet}
</DatePicker.Root>Range Date Picker
<script>
  import {DatePicker} from 'ui-ingredients';
  import {
    CalendarIcon,
    ChevronLeftIcon,
    ChevronRightIcon,
    XCloseIcon,
  } from '$lib/icons';
</script>
<DatePicker.Root fixedWeeks selectionMode="range">
  {#snippet children(context)}
    <DatePicker.Label>Label</DatePicker.Label>
    <DatePicker.Control>
      <DatePicker.Input index={0} />
      <DatePicker.Input index={1} />
      <DatePicker.ClearTrigger>
        <XCloseIcon />
      </DatePicker.ClearTrigger>
      <DatePicker.Trigger>
        <CalendarIcon />
      </DatePicker.Trigger>
    </DatePicker.Control>
    <DatePicker.Positioner>
      <DatePicker.Content>
        <DatePicker.View view="day">
          <DatePicker.ViewControl>
            <DatePicker.PrevTrigger>
              <ChevronLeftIcon />
            </DatePicker.PrevTrigger>
            <DatePicker.ViewTrigger>
              <DatePicker.RangeText />
            </DatePicker.ViewTrigger>
            <DatePicker.NextTrigger>
              <ChevronRightIcon />
            </DatePicker.NextTrigger>
          </DatePicker.ViewControl>
          <DatePicker.Table>
            <DatePicker.TableHead>
              <DatePicker.TableRow>
                {#each context.weekDays as weekDay}
                  <DatePicker.TableHeader>
                    {weekDay.narrow}
                  </DatePicker.TableHeader>
                {/each}
              </DatePicker.TableRow>
            </DatePicker.TableHead>
            <DatePicker.TableBody>
              {#each context.weeks as week}
                <DatePicker.TableRow>
                  {#each week as day}
                    <DatePicker.DayTableCell value={day}>
                      <DatePicker.DayTableCellTrigger>
                        {day.day}
                      </DatePicker.DayTableCellTrigger>
                    </DatePicker.DayTableCell>
                  {/each}
                </DatePicker.TableRow>
              {/each}
            </DatePicker.TableBody>
          </DatePicker.Table>
        </DatePicker.View>
        <!-- MONTH -->
        <DatePicker.View view="month">
          <DatePicker.ViewControl>
            <DatePicker.PrevTrigger>
              <ChevronLeftIcon />
            </DatePicker.PrevTrigger>
            <DatePicker.ViewTrigger>
              <DatePicker.RangeText />
            </DatePicker.ViewTrigger>
            <DatePicker.NextTrigger>
              <ChevronRightIcon />
            </DatePicker.NextTrigger>
          </DatePicker.ViewControl>
          <DatePicker.Table>
            <DatePicker.TableBody>
              {#each context.getMonthsGrid( {columns: 4, format: 'short'}, ) as months}
                <DatePicker.TableRow>
                  {#each months as month}
                    <DatePicker.MonthTableCell value={month.value}>
                      <DatePicker.MonthTableCellTrigger>
                        {month.label}
                      </DatePicker.MonthTableCellTrigger>
                    </DatePicker.MonthTableCell>
                  {/each}
                </DatePicker.TableRow>
              {/each}
            </DatePicker.TableBody>
          </DatePicker.Table>
        </DatePicker.View>
        <!-- YEAR -->
        <DatePicker.View view="year">
          <DatePicker.ViewControl>
            <DatePicker.PrevTrigger>
              <ChevronLeftIcon />
            </DatePicker.PrevTrigger>
            <DatePicker.ViewTrigger>
              <DatePicker.RangeText />
            </DatePicker.ViewTrigger>
            <DatePicker.NextTrigger>
              <ChevronRightIcon />
            </DatePicker.NextTrigger>
          </DatePicker.ViewControl>
          <DatePicker.Table>
            <DatePicker.TableBody>
              {#each context.getYearsGrid({columns: 4}) as years}
                <DatePicker.TableRow>
                  {#each years as year}
                    <DatePicker.YearTableCell value={year.value}>
                      <DatePicker.YearTableCellTrigger>
                        {year.label}
                      </DatePicker.YearTableCellTrigger>
                    </DatePicker.YearTableCell>
                  {/each}
                </DatePicker.TableRow>
              {/each}
            </DatePicker.TableBody>
          </DatePicker.Table>
        </DatePicker.View>
      </DatePicker.Content>
    </DatePicker.Positioner>
  {/snippet}
</DatePicker.Root>API Reference
Root
| Prop | Default | Description | 
|---|---|---|
| closeOnSelect | true | booleanWhether the calendar should close after the date selection is complete.
This is ignored when the selection mode is `multiple`. | 
| defaultFocusedValue | DateValueThe initial focused date when rendered.
Use when you don't need to control the focused date of the date picker. | |
| defaultOpen | booleanThe initial open state of the date picker when rendered.
Use when you don't need to control the open state of the date picker. | |
| defaultValue | DateValue[]The initial selected date(s) when rendered.
Use when you don't need to control the selected date(s) of the date picker. | |
| defaultView | "day" | DateViewThe default view of the calendar | 
| disabled | booleanWhether the calendar is disabled. | |
| fixedWeeks | booleanWhether the calendar should have a fixed number of weeks.
This renders the calendar with 6 weeks instead of 5 or 6. | |
| focusedValue | DateValueThe controlled focused date. | |
| format | (date: LocaleDetails) => stringThe format of the date to display in the input. | |
| id | stringThe unique identifier of the machine. | |
| ids | {
	root?: string;
	label(index: number)?: string;
	table(id: string)?: string;
	tableHeader(id: string)?: string;
	tableBody(id: string)?: string;
	tableRow(id: string)?: string;
	content?: string;
	cellTrigger(id: string)?: string;
	prevTrigger(view: DateView)?: string;
	clearTrigger: string;
	control: string;
	input(index: number)?: string;
	trigger?: string;
	monthSelect?: string;
	yearSelect?: string;
	positioner?: string;
}The ids of the elements in the date picker. Useful for composition. | |
| isDateUnavailable | (date: DateValue, locale: string) => booleanReturns whether a date of the calendar is available. | |
| keepMounted | booleanWhether to keep the component mounted after exit. | |
| lazyMount | booleanWhether to enable lazy mounting. | |
| locale | "en-US" | stringThe locale (BCP 47 language tag) to use when formatting the date. | 
| max | DateValueThe maximum date that can be selected. | |
| maxView | "year" | DateViewThe maximum view of the calendar | 
| min | DateValueThe minimum date that can be selected. | |
| minView | "day" | DateViewThe minimum view of the calendar | 
| name | stringThe `name` attribute of the input element. | |
| numOfMonths | numberThe number of months to display. | |
| onFocusChange | (details: FocusChangeDetails) => voidFunction called when the focused date changes. | |
| onOpenChange | (details: OpenChangeDetails) => voidFunction called when the calendar opens or closes. | |
| onValueChange | (details: ValueChangeDetails) => voidFunction called when the value changes. | |
| onViewChange | (details: ViewChangeDetails) => voidFunction called when the view changes. | |
| open | booleanThe controlled open state of the date picker | |
| outsideDaySelectable | false | booleanWhether day outside the visible range can be selected. | 
| parse | (value: string, details: LocaleDetails) => DateValueFunction to parse the date from the input back to a DateValue. | |
| placeholder | stringThe placeholder text to display in the input. | |
| positioning | PositioningOptionsThe user provided options used to position the date picker content | |
| readOnly | booleanWhether the calendar is read-only. | |
| selectionMode | "single" | SelectionModeThe selection mode of the calendar.
- `single` - only one date can be selected
- `multiple` - multiple dates can be selected
- `range` - a range of dates can be selected | 
| startOfWeek | numberThe first day of the week.
 `0` - Sunday
 `1` - Monday
 `2` - Tuesday
 `3` - Wednesday
 `4` - Thursday
 `5` - Friday
 `6` - Saturday | |
| timeZone | "UTC" | stringThe time zone to use | 
| translations | IntlTranslationsThe localized messages to use. | |
| value | DateValue[]The controlled selected date(s). | |
| view | DateViewThe view of the calendar | |
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-part | root | 
| data-readonly | Present when read-only | 
| data-scope | date-picker | 
| data-state | "open" | "closed" | 
ClearTrigger
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
Content
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-part | content | 
| data-placement | The placement of the content | 
| data-scope | date-picker | 
| data-state | "open" | "closed" | 
Control
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-part | control | 
| data-scope | date-picker | 
Input
| Prop | Default | Description | 
|---|---|---|
| index | number | |
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-index | The index of the item | 
| data-part | input | 
| data-scope | date-picker | 
| data-state | "open" | "closed" | 
Label
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-index | The index of the item | 
| data-part | label | 
| data-readonly | Present when read-only | 
| data-scope | date-picker | 
| data-state | "open" | "closed" | 
MonthSelect
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
NextTrigger
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-part | next-trigger | 
| data-scope | date-picker | 
Positioner
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
PresetTrigger
| Prop | Default | Description | 
|---|---|---|
| value | PresetTriggerValue | |
| asChild | SnippetRender a different element. | 
PrevTrigger
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-part | prev-trigger | 
| data-scope | date-picker | 
RangeText
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
Table
| Prop | Default | Description | 
|---|---|---|
| columns | number | |
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-columns | |
| data-part | table | 
| data-scope | date-picker | 
| data-view | The view of the table | 
TableBody
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-part | table-body | 
| data-scope | date-picker | 
| data-view | The view of the tablebody | 
TableCell
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
TableCellTrigger
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
TableHead
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-part | table-head | 
| data-scope | date-picker | 
| data-view | The view of the tablehead | 
TableHeader
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-part | table-header | 
| data-scope | date-picker | 
| data-view | The view of the tableheader | 
TableRow
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-disabled | Present when disabled | 
| data-part | table-row | 
| data-scope | date-picker | 
| data-view | The view of the tablerow | 
Trigger
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-part | trigger | 
| data-placement | The placement of the trigger | 
| data-scope | date-picker | 
| data-state | "open" | "closed" | 
View
| Prop | Default | Description | 
|---|---|---|
| view | DateView | |
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-part | view | 
| data-scope | date-picker | 
| data-view | The view of the view | 
ViewControl
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-part | view-control | 
| data-scope | date-picker | 
| data-view | The view of the viewcontrol | 
ViewTrigger
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
| Data Attribute | Value | 
|---|---|
| data-part | view-trigger | 
| data-scope | date-picker | 
| data-view | The view of the viewtrigger | 
YearSelect
| Prop | Default | Description | 
|---|---|---|
| asChild | SnippetRender a different element. | 
Accessibility
Keyboard Support
| Key | Description | 
|---|---|
| ArrowLeft | Moves focus to the previous day within the current week. | 
| ArrowRight | Moves focus to the next day within the current week. | 
| ArrowUp | Moves focus to the same day of the week in the previous week. | 
| ArrowDown | Moves focus to the same day of the week in the next week. | 
| Home | Moves focus to the first day of the current month. | 
| End | Moves focus to the last day of the current month. | 
| PageUp | Moves focus to the same day of the month in the previous month. | 
| PageDown | Moves focus to the same day of the month in the next month. | 
| Enter | Selects the focused date and closes the date picker. | 
| Esc | Closes the date picker without selecting any date. |