@codecademy/gamut 68.1.1-alpha.613ffe.0 → 68.1.1-alpha.9135d1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (40) hide show
  1. package/dist/ConnectedForm/utils.d.ts +1 -1
  2. package/dist/DatePicker/Calendar/Calendar.d.ts +11 -0
  3. package/dist/DatePicker/Calendar/Calendar.js +16 -0
  4. package/dist/DatePicker/Calendar/CalendarBody.d.ts +3 -0
  5. package/dist/DatePicker/Calendar/CalendarBody.js +219 -0
  6. package/dist/DatePicker/Calendar/CalendarFooter.d.ts +3 -0
  7. package/dist/DatePicker/Calendar/CalendarFooter.js +60 -0
  8. package/dist/DatePicker/Calendar/CalendarHeader.d.ts +3 -0
  9. package/dist/DatePicker/Calendar/CalendarHeader.js +51 -0
  10. package/dist/DatePicker/Calendar/index.d.ts +6 -0
  11. package/dist/DatePicker/Calendar/index.js +5 -0
  12. package/dist/DatePicker/Calendar/types.d.ts +57 -0
  13. package/dist/DatePicker/Calendar/types.js +1 -0
  14. package/dist/DatePicker/Calendar/utils/dateGrid.d.ts +34 -0
  15. package/dist/DatePicker/Calendar/utils/dateGrid.js +101 -0
  16. package/dist/DatePicker/Calendar/utils/format.d.ts +39 -0
  17. package/dist/DatePicker/Calendar/utils/format.js +130 -0
  18. package/dist/DatePicker/Calendar/utils/index.d.ts +3 -0
  19. package/dist/DatePicker/Calendar/utils/index.js +3 -0
  20. package/dist/DatePicker/Calendar/utils/validation.d.ts +13 -0
  21. package/dist/DatePicker/Calendar/utils/validation.js +23 -0
  22. package/dist/DatePicker/DatePicker.d.ts +8 -0
  23. package/dist/DatePicker/DatePicker.js +125 -0
  24. package/dist/DatePicker/DatePickerCalendar.d.ts +13 -0
  25. package/dist/DatePicker/DatePickerCalendar.js +160 -0
  26. package/dist/DatePicker/DatePickerContext.d.ts +11 -0
  27. package/dist/DatePicker/DatePickerContext.js +18 -0
  28. package/dist/DatePicker/DatePickerInput.d.ts +16 -0
  29. package/dist/DatePicker/DatePickerInput.js +127 -0
  30. package/dist/DatePicker/index.d.ts +13 -0
  31. package/dist/DatePicker/index.js +10 -0
  32. package/dist/DatePicker/types.d.ts +93 -0
  33. package/dist/DatePicker/types.js +1 -0
  34. package/dist/FocusTrap/index.d.ts +2 -2
  35. package/dist/List/elements.d.ts +1 -1
  36. package/dist/PopoverContainer/PopoverContainer.js +3 -1
  37. package/dist/PopoverContainer/types.d.ts +5 -0
  38. package/dist/index.d.ts +1 -0
  39. package/dist/index.js +1 -0
  40. package/package.json +6 -6
@@ -122,9 +122,9 @@ export declare function useDebouncedField<T extends InputTypes>({ name, watchUpd
122
122
  value: T extends "checkbox" ? boolean : string;
123
123
  error: string | FieldError | Merge<FieldError, FieldErrorsImpl<any>> | undefined;
124
124
  ref: import("react-hook-form").UseFormRegisterReturn<string>;
125
- control: import("react-hook-form").Control<import("react-hook-form").FieldValues, any, import("react-hook-form").FieldValues>;
126
125
  isDisabled: boolean;
127
126
  isLoading: boolean | undefined;
127
+ control: import("react-hook-form").Control<import("react-hook-form").FieldValues, any, import("react-hook-form").FieldValues>;
128
128
  isSoloField: boolean | undefined;
129
129
  validation: RegisterOptions | undefined;
130
130
  getValues: import("react-hook-form").UseFormGetValues<import("react-hook-form").FieldValues>;
@@ -0,0 +1,11 @@
1
+ /// <reference types="react" />
2
+ /**
3
+ * Outer wrapper for the calendar (header + body + footer).
4
+ * Used by DatePickerCalendar to group the calendar content.
5
+ */
6
+ export declare const Calendar: import("@emotion/styled").StyledComponent<{
7
+ theme?: import("@emotion/react").Theme | undefined;
8
+ as?: import("react").ElementType<any, keyof import("react").JSX.IntrinsicElements> | undefined;
9
+ } & {
10
+ theme?: import("@emotion/react").Theme | undefined;
11
+ }, import("react").DetailedHTMLProps<import("react").HTMLAttributes<HTMLDivElement>, HTMLDivElement>, {}>;
@@ -0,0 +1,16 @@
1
+ import _styled from "@emotion/styled/base";
2
+ import { css } from '@codecademy/gamut-styles';
3
+ /**
4
+ * Outer wrapper for the calendar (header + body + footer).
5
+ * Used by DatePickerCalendar to group the calendar content.
6
+ */
7
+ export const Calendar = /*#__PURE__*/_styled("div", {
8
+ target: "eq0g1i20",
9
+ label: "Calendar"
10
+ })(css({
11
+ backgroundColor: 'background',
12
+ borderRadius: 'lg',
13
+ boxShadow: '0 4px 16px rgba(0, 0, 0, 0.12)',
14
+ width: 'max-content',
15
+ p: 24
16
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9EYXRlUGlja2VyL0NhbGVuZGFyL0NhbGVuZGFyLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFPd0IiLCJmaWxlIjoiLi4vLi4vLi4vc3JjL0RhdGVQaWNrZXIvQ2FsZW5kYXIvQ2FsZW5kYXIudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQtc3R5bGVzJztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcblxuLyoqXG4gKiBPdXRlciB3cmFwcGVyIGZvciB0aGUgY2FsZW5kYXIgKGhlYWRlciArIGJvZHkgKyBmb290ZXIpLlxuICogVXNlZCBieSBEYXRlUGlja2VyQ2FsZW5kYXIgdG8gZ3JvdXAgdGhlIGNhbGVuZGFyIGNvbnRlbnQuXG4gKi9cbmV4cG9ydCBjb25zdCBDYWxlbmRhciA9IHN0eWxlZC5kaXYoXG4gIGNzcyh7XG4gICAgYmFja2dyb3VuZENvbG9yOiAnYmFja2dyb3VuZCcsXG4gICAgYm9yZGVyUmFkaXVzOiAnbGcnLFxuICAgIGJveFNoYWRvdzogJzAgNHB4IDE2cHggcmdiYSgwLCAwLCAwLCAwLjEyKScsXG4gICAgd2lkdGg6ICdtYXgtY29udGVudCcsXG4gICAgcDogMjQsXG4gIH0pXG4pO1xuIl19 */");
@@ -0,0 +1,3 @@
1
+ import * as React from 'react';
2
+ import { CalendarBodyProps } from './types';
3
+ export declare const CalendarBody: React.FC<CalendarBodyProps>;
@@ -0,0 +1,219 @@
1
+ import _styled from "@emotion/styled/base";
2
+ import { states } from '@codecademy/gamut-styles';
3
+ import { useCallback, useEffect, useMemo, useRef } from 'react';
4
+ import * as React from 'react';
5
+ import { TextButton } from '../../Button';
6
+ import { clampToMonth, getMonthGrid, isDateDisabled, isDateInRange, isSameDay } from './utils/dateGrid';
7
+ import { getWeekdayFullNames, getWeekdayLabels } from './utils/format';
8
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
9
+ const DateButton = /*#__PURE__*/_styled(TextButton, {
10
+ target: "e12sl4cx0",
11
+ label: "DateButton"
12
+ })(states({
13
+ isToday: {
14
+ bg: 'navy-200'
15
+ },
16
+ isSelected: {
17
+ color: 'background',
18
+ bg: 'text'
19
+ },
20
+ isInRange: {
21
+ bg: 'border-secondary'
22
+ }
23
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DatePicker/Calendar/CalendarBody.tsx"],"names":[],"mappings":"AAgBmB","file":"../../../src/DatePicker/Calendar/CalendarBody.tsx","sourcesContent":["import { states } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { useCallback, useEffect, useMemo, useRef } from 'react';\nimport * as React from 'react';\n\nimport { TextButton } from '../../Button';\nimport { CalendarBodyProps } from './types';\nimport {\n  clampToMonth,\n  getMonthGrid,\n  isDateDisabled,\n  isDateInRange,\n  isSameDay,\n} from './utils/dateGrid';\nimport { getWeekdayFullNames, getWeekdayLabels } from './utils/format';\n\nconst DateButton = styled(TextButton)(\n  states({\n    isToday: {\n      bg: 'navy-200',\n    },\n    isSelected: {\n      color: 'background',\n      bg: 'text',\n    },\n    isInRange: {\n      bg: 'border-secondary',\n    },\n  })\n);\n\n/** Flat list of dates in grid order (row-major, non-null only) with row index for Home/End */\nfunction getDatesWithRow(\n  weeks: (Date | null)[][]\n): { date: Date; rowIndex: number }[] {\n  const result: { date: Date; rowIndex: number }[] = [];\n  weeks.forEach((week, rowIndex) => {\n    week.forEach((date) => {\n      if (date !== null) result.push({ date, rowIndex });\n    });\n  });\n  return result;\n}\n\nexport const CalendarBody: React.FC<CalendarBodyProps> = ({\n  visibleDate,\n  selectedDate,\n  endDate = null,\n  disabledDates = [],\n  onDateSelect,\n  locale,\n  weekStartsOn = 0,\n  labelledById,\n  focusedDate,\n  onFocusedDateChange,\n  onVisibleDateChange,\n  onEscapeKeyPress,\n}) => {\n  const year = visibleDate.getFullYear();\n  const month = visibleDate.getMonth();\n  const weeks = getMonthGrid(year, month, weekStartsOn);\n  const weekdayLabels = getWeekdayLabels(locale, weekStartsOn);\n  const weekdayFullNames = getWeekdayFullNames(locale, weekStartsOn);\n  const buttonRefs = useRef<Map<number, HTMLElement>>(new Map());\n\n  const datesWithRow = useMemo(() => getDatesWithRow(weeks), [weeks]);\n  const focusTarget = focusedDate ?? selectedDate;\n\n  const isToday = useCallback(\n    (d: Date | null) => d !== null && isSameDay(d, new Date()),\n    []\n  );\n\n  const focusButton = useCallback((date: Date | null) => {\n    if (date === null) return;\n    const key = new Date(\n      date.getFullYear(),\n      date.getMonth(),\n      date.getDate()\n    ).getTime();\n    buttonRefs.current.get(key)?.focus();\n  }, []);\n\n  useEffect(() => {\n    if (focusTarget !== null) focusButton(focusTarget);\n  }, [focusTarget, focusButton]);\n\n  const handleKeyDown = useCallback(\n    (e: React.KeyboardEvent, date: Date) => {\n      if (!onFocusedDateChange) return;\n      const key = date.getTime();\n      const idx = datesWithRow.findIndex(({ date: d }) => d.getTime() === key);\n      if (idx < 0) return;\n\n      const currentRow = datesWithRow[idx].rowIndex;\n      const day = date.getDate();\n\n      let newDate: Date | null = null;\n      let newVisibleDate: Date | null = null;\n\n      switch (e.key) {\n        case 'ArrowLeft':\n          e.preventDefault();\n          if (idx > 0) {\n            newDate = datesWithRow[idx - 1].date;\n          } else {\n            const lastDayPrevMonth = new Date(year, month, 0);\n            newDate = lastDayPrevMonth;\n            newVisibleDate = new Date(year, month - 1, 1);\n          }\n          break;\n        case 'ArrowRight':\n          e.preventDefault();\n          if (idx < datesWithRow.length - 1) {\n            newDate = datesWithRow[idx + 1].date;\n          } else {\n            newDate = new Date(year, month + 1, 1);\n            newVisibleDate = new Date(year, month + 1, 1);\n          }\n          break;\n        case 'ArrowUp':\n          e.preventDefault();\n          newDate = new Date(date);\n          newDate.setDate(newDate.getDate() - 7);\n          if (newDate.getMonth() !== month || newDate.getFullYear() !== year) {\n            newVisibleDate = new Date(\n              newDate.getFullYear(),\n              newDate.getMonth(),\n              1\n            );\n          }\n          break;\n        case 'ArrowDown':\n          e.preventDefault();\n          newDate = new Date(date);\n          newDate.setDate(newDate.getDate() + 7);\n          if (newDate.getMonth() !== month || newDate.getFullYear() !== year) {\n            newVisibleDate = new Date(\n              newDate.getFullYear(),\n              newDate.getMonth(),\n              1\n            );\n          }\n          break;\n        case 'Home':\n          e.preventDefault();\n          newDate =\n            datesWithRow.find(({ rowIndex }) => rowIndex === currentRow)\n              ?.date ?? date;\n          break;\n        case 'End':\n          e.preventDefault();\n          newDate =\n            [...datesWithRow]\n              .reverse()\n              .find(({ rowIndex }) => rowIndex === currentRow)?.date ?? date;\n          break;\n        case 'PageDown':\n          e.preventDefault();\n          if (e.shiftKey) {\n            newDate = clampToMonth(year + 1, month, day);\n          } else {\n            newDate = clampToMonth(year, month + 1, day);\n          }\n          newVisibleDate = new Date(\n            newDate.getFullYear(),\n            newDate.getMonth(),\n            1\n          );\n          break;\n        case 'PageUp':\n          e.preventDefault();\n          if (e.shiftKey) {\n            newDate = clampToMonth(year - 1, month, day);\n          } else {\n            newDate = clampToMonth(year, month - 1, day);\n          }\n          newVisibleDate = new Date(\n            newDate.getFullYear(),\n            newDate.getMonth(),\n            1\n          );\n          break;\n        case 'Enter':\n        case ' ':\n          e.preventDefault();\n          if (!isDateDisabled(date, disabledDates)) onDateSelect(date);\n          return;\n        case 'Escape':\n          e.preventDefault();\n          onEscapeKeyPress?.();\n          return;\n        default:\n          return;\n      }\n\n      if (newDate !== null) {\n        onFocusedDateChange(newDate);\n        if (newVisibleDate !== null) onVisibleDateChange?.(newVisibleDate);\n      }\n    },\n    [\n      onFocusedDateChange,\n      datesWithRow,\n      month,\n      year,\n      disabledDates,\n      onDateSelect,\n      onEscapeKeyPress,\n      onVisibleDateChange,\n    ]\n  );\n\n  const setButtonRef = useCallback((date: Date, el: HTMLElement | null) => {\n    const k = new Date(\n      date.getFullYear(),\n      date.getMonth(),\n      date.getDate()\n    ).getTime();\n    if (el) buttonRefs.current.set(k, el);\n    else buttonRefs.current.delete(k);\n  }, []);\n\n  return (\n    <table aria-labelledby={labelledById} role=\"grid\" width=\"100%\">\n      <thead>\n        <tr>\n          {weekdayLabels.map((label, i) => (\n            <th abbr={weekdayFullNames[i]} key={label} scope=\"col\">\n              {label}\n            </th>\n          ))}\n        </tr>\n      </thead>\n      <tbody>\n        {weeks.map((week, rowIndex) => (\n          <tr key={week.join('-')}>\n            {week.map((date, colIndex) => {\n              if (date === null) {\n                return (\n                  // fix this error\n                  // eslint-disable-next-line react/no-array-index-key, jsx-a11y/control-has-associated-label\n                  <td key={`empty-${rowIndex}-${colIndex}`} role=\"gridcell\" />\n                );\n              }\n              const selected =\n                isSameDay(date, selectedDate) || isSameDay(date, endDate);\n              const inRange =\n                (selectedDate !== null || endDate !== null) &&\n                isDateInRange(date, selectedDate, endDate);\n              const disabled = isDateDisabled(date, disabledDates);\n              const today = isToday(date);\n              const isFocused =\n                focusTarget !== null && isSameDay(date, focusTarget);\n\n              return (\n                <td\n                  aria-selected={selected}\n                  key={date.getTime()}\n                  role=\"gridcell\"\n                >\n                  <DateButton\n                    disabled={disabled}\n                    isInRange={inRange}\n                    isSelected={selected}\n                    isToday={today}\n                    ref={(el) => setButtonRef(date, el as HTMLElement | null)}\n                    tabIndex={isFocused ? 0 : -1}\n                    variant=\"secondary\"\n                    width=\"36px\"\n                    onClick={() => onDateSelect(date)}\n                    onFocus={() => onFocusedDateChange?.(date)}\n                    onKeyDown={(e: React.KeyboardEvent) =>\n                      handleKeyDown(e, date)\n                    }\n                  >\n                    {date.getDate()}\n                  </DateButton>\n                </td>\n              );\n            })}\n          </tr>\n        ))}\n      </tbody>\n    </table>\n  );\n};\n"]} */");
24
+
25
+ /** Flat list of dates in grid order (row-major, non-null only) with row index for Home/End */
26
+ function getDatesWithRow(weeks) {
27
+ const result = [];
28
+ weeks.forEach((week, rowIndex) => {
29
+ week.forEach(date => {
30
+ if (date !== null) result.push({
31
+ date,
32
+ rowIndex
33
+ });
34
+ });
35
+ });
36
+ return result;
37
+ }
38
+ export const CalendarBody = ({
39
+ visibleDate,
40
+ selectedDate,
41
+ endDate = null,
42
+ disabledDates = [],
43
+ onDateSelect,
44
+ locale,
45
+ weekStartsOn = 0,
46
+ labelledById,
47
+ focusedDate,
48
+ onFocusedDateChange,
49
+ onVisibleDateChange,
50
+ onEscapeKeyPress
51
+ }) => {
52
+ const year = visibleDate.getFullYear();
53
+ const month = visibleDate.getMonth();
54
+ const weeks = getMonthGrid(year, month, weekStartsOn);
55
+ const weekdayLabels = getWeekdayLabels(locale, weekStartsOn);
56
+ const weekdayFullNames = getWeekdayFullNames(locale, weekStartsOn);
57
+ const buttonRefs = useRef(new Map());
58
+ const datesWithRow = useMemo(() => getDatesWithRow(weeks), [weeks]);
59
+ const focusTarget = focusedDate ?? selectedDate;
60
+ const isToday = useCallback(d => d !== null && isSameDay(d, new Date()), []);
61
+ const focusButton = useCallback(date => {
62
+ if (date === null) return;
63
+ const key = new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime();
64
+ buttonRefs.current.get(key)?.focus();
65
+ }, []);
66
+ useEffect(() => {
67
+ if (focusTarget !== null) focusButton(focusTarget);
68
+ }, [focusTarget, focusButton]);
69
+ const handleKeyDown = useCallback((e, date) => {
70
+ if (!onFocusedDateChange) return;
71
+ const key = date.getTime();
72
+ const idx = datesWithRow.findIndex(({
73
+ date: d
74
+ }) => d.getTime() === key);
75
+ if (idx < 0) return;
76
+ const currentRow = datesWithRow[idx].rowIndex;
77
+ const day = date.getDate();
78
+ let newDate = null;
79
+ let newVisibleDate = null;
80
+ switch (e.key) {
81
+ case 'ArrowLeft':
82
+ e.preventDefault();
83
+ if (idx > 0) {
84
+ newDate = datesWithRow[idx - 1].date;
85
+ } else {
86
+ const lastDayPrevMonth = new Date(year, month, 0);
87
+ newDate = lastDayPrevMonth;
88
+ newVisibleDate = new Date(year, month - 1, 1);
89
+ }
90
+ break;
91
+ case 'ArrowRight':
92
+ e.preventDefault();
93
+ if (idx < datesWithRow.length - 1) {
94
+ newDate = datesWithRow[idx + 1].date;
95
+ } else {
96
+ newDate = new Date(year, month + 1, 1);
97
+ newVisibleDate = new Date(year, month + 1, 1);
98
+ }
99
+ break;
100
+ case 'ArrowUp':
101
+ e.preventDefault();
102
+ newDate = new Date(date);
103
+ newDate.setDate(newDate.getDate() - 7);
104
+ if (newDate.getMonth() !== month || newDate.getFullYear() !== year) {
105
+ newVisibleDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
106
+ }
107
+ break;
108
+ case 'ArrowDown':
109
+ e.preventDefault();
110
+ newDate = new Date(date);
111
+ newDate.setDate(newDate.getDate() + 7);
112
+ if (newDate.getMonth() !== month || newDate.getFullYear() !== year) {
113
+ newVisibleDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
114
+ }
115
+ break;
116
+ case 'Home':
117
+ e.preventDefault();
118
+ newDate = datesWithRow.find(({
119
+ rowIndex
120
+ }) => rowIndex === currentRow)?.date ?? date;
121
+ break;
122
+ case 'End':
123
+ e.preventDefault();
124
+ newDate = [...datesWithRow].reverse().find(({
125
+ rowIndex
126
+ }) => rowIndex === currentRow)?.date ?? date;
127
+ break;
128
+ case 'PageDown':
129
+ e.preventDefault();
130
+ if (e.shiftKey) {
131
+ newDate = clampToMonth(year + 1, month, day);
132
+ } else {
133
+ newDate = clampToMonth(year, month + 1, day);
134
+ }
135
+ newVisibleDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
136
+ break;
137
+ case 'PageUp':
138
+ e.preventDefault();
139
+ if (e.shiftKey) {
140
+ newDate = clampToMonth(year - 1, month, day);
141
+ } else {
142
+ newDate = clampToMonth(year, month - 1, day);
143
+ }
144
+ newVisibleDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
145
+ break;
146
+ case 'Enter':
147
+ case ' ':
148
+ e.preventDefault();
149
+ if (!isDateDisabled(date, disabledDates)) onDateSelect(date);
150
+ return;
151
+ case 'Escape':
152
+ e.preventDefault();
153
+ onEscapeKeyPress?.();
154
+ return;
155
+ default:
156
+ return;
157
+ }
158
+ if (newDate !== null) {
159
+ onFocusedDateChange(newDate);
160
+ if (newVisibleDate !== null) onVisibleDateChange?.(newVisibleDate);
161
+ }
162
+ }, [onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onVisibleDateChange]);
163
+ const setButtonRef = useCallback((date, el) => {
164
+ const k = new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime();
165
+ if (el) buttonRefs.current.set(k, el);else buttonRefs.current.delete(k);
166
+ }, []);
167
+ return /*#__PURE__*/_jsxs("table", {
168
+ "aria-labelledby": labelledById,
169
+ role: "grid",
170
+ width: "100%",
171
+ children: [/*#__PURE__*/_jsx("thead", {
172
+ children: /*#__PURE__*/_jsx("tr", {
173
+ children: weekdayLabels.map((label, i) => /*#__PURE__*/_jsx("th", {
174
+ abbr: weekdayFullNames[i],
175
+ scope: "col",
176
+ children: label
177
+ }, label))
178
+ })
179
+ }), /*#__PURE__*/_jsx("tbody", {
180
+ children: weeks.map((week, rowIndex) => /*#__PURE__*/_jsx("tr", {
181
+ children: week.map((date, colIndex) => {
182
+ if (date === null) {
183
+ return (
184
+ /*#__PURE__*/
185
+ // fix this error
186
+ // eslint-disable-next-line react/no-array-index-key, jsx-a11y/control-has-associated-label
187
+ _jsx("td", {
188
+ role: "gridcell"
189
+ }, `empty-${rowIndex}-${colIndex}`)
190
+ );
191
+ }
192
+ const selected = isSameDay(date, selectedDate) || isSameDay(date, endDate);
193
+ const inRange = (selectedDate !== null || endDate !== null) && isDateInRange(date, selectedDate, endDate);
194
+ const disabled = isDateDisabled(date, disabledDates);
195
+ const today = isToday(date);
196
+ const isFocused = focusTarget !== null && isSameDay(date, focusTarget);
197
+ return /*#__PURE__*/_jsx("td", {
198
+ "aria-selected": selected,
199
+ role: "gridcell",
200
+ children: /*#__PURE__*/_jsx(DateButton, {
201
+ disabled: disabled,
202
+ isInRange: inRange,
203
+ isSelected: selected,
204
+ isToday: today,
205
+ ref: el => setButtonRef(date, el),
206
+ tabIndex: isFocused ? 0 : -1,
207
+ variant: "secondary",
208
+ width: "36px",
209
+ onClick: () => onDateSelect(date),
210
+ onFocus: () => onFocusedDateChange?.(date),
211
+ onKeyDown: e => handleKeyDown(e, date),
212
+ children: date.getDate()
213
+ })
214
+ }, date.getTime());
215
+ })
216
+ }, week.join('-')))
217
+ })]
218
+ });
219
+ };
@@ -0,0 +1,3 @@
1
+ import * as React from 'react';
2
+ import { CalendarFooterProps } from './types';
3
+ export declare const CalendarFooter: React.FC<CalendarFooterProps>;
@@ -0,0 +1,60 @@
1
+ import * as React from 'react';
2
+ import { FlexBox } from '../../Box';
3
+ import { TextButton } from '../../Button';
4
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
5
+ // function formatQuickActionLabel(action: QuickAction): string {
6
+ // const { num, timePeriod } = action;
7
+ // const period =
8
+ // timePeriod === 'day'
9
+ // ? num === 1
10
+ // ? 'day'
11
+ // : 'days'
12
+ // : timePeriod === 'week'
13
+ // ? num === 1
14
+ // ? 'week'
15
+ // : 'weeks'
16
+ // : timePeriod === 'month'
17
+ // ? num === 1
18
+ // ? 'month'
19
+ // : 'months'
20
+ // : num === 1
21
+ // ? 'year'
22
+ // : 'years';
23
+ // return `${num} ${period}`;
24
+ // }
25
+
26
+ export const CalendarFooter = ({
27
+ onClearDate,
28
+ onTodayClick,
29
+ onSelectedDateChange,
30
+ onCurrentMonthYearChange
31
+ }) => {
32
+ const handleClearDate = () => {
33
+ onSelectedDateChange(null);
34
+ onClearDate?.();
35
+ };
36
+ const handleTodayClick = () => {
37
+ const today = new Date();
38
+ onSelectedDateChange(today);
39
+ onCurrentMonthYearChange(new Date(today.getFullYear(), today.getMonth(), 1));
40
+ onTodayClick?.();
41
+ };
42
+ // const actions = quickActions.slice(0, 3);
43
+
44
+ return /*#__PURE__*/_jsxs(FlexBox, {
45
+ alignItems: "center",
46
+ borderTop: 1,
47
+ justifyContent: "space-between",
48
+ py: 12,
49
+ children: [/*#__PURE__*/_jsx(TextButton, {
50
+ onClick: handleClearDate,
51
+ children: "Clear"
52
+ }), /*#__PURE__*/_jsx(FlexBox, {
53
+ gap: 32,
54
+ children: /*#__PURE__*/_jsx(TextButton, {
55
+ onClick: handleTodayClick,
56
+ children: "Today"
57
+ })
58
+ })]
59
+ });
60
+ };
@@ -0,0 +1,3 @@
1
+ import * as React from 'react';
2
+ import { CalendarHeaderProps } from './types';
3
+ export declare const CalendarHeader: React.FC<CalendarHeaderProps>;
@@ -0,0 +1,51 @@
1
+ import { MiniChevronLeftIcon, MiniChevronRightIcon } from '@codecademy/gamut-icons';
2
+ import * as React from 'react';
3
+ import { FlexBox } from '../../Box';
4
+ import { IconButton } from '../../Button';
5
+ import { Text } from '../../Typography';
6
+ import { formatMonthYear } from './utils/format';
7
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
8
+ export const CalendarHeader = ({
9
+ currentMonthYear,
10
+ onCurrentMonthYearChange,
11
+ onPreviousMonthClick,
12
+ onNextMonthClick,
13
+ locale,
14
+ headingId
15
+ }) => {
16
+ const monthYear = formatMonthYear(currentMonthYear, locale);
17
+ const handlePreviousMonth = () => {
18
+ const previousMonth = new Date(currentMonthYear.getFullYear(), currentMonthYear.getMonth() - 1, 1);
19
+ onCurrentMonthYearChange?.(previousMonth);
20
+ onPreviousMonthClick?.();
21
+ };
22
+ const handleNextMonth = () => {
23
+ const nextMonth = new Date(currentMonthYear.getFullYear(), currentMonthYear.getMonth() + 1, 1);
24
+ onCurrentMonthYearChange?.(nextMonth);
25
+ onNextMonthClick?.();
26
+ };
27
+ return /*#__PURE__*/_jsxs(FlexBox, {
28
+ alignItems: "center",
29
+ justifyContent: "space-between",
30
+ pb: 16,
31
+ children: [/*#__PURE__*/_jsx(IconButton, {
32
+ "aria-label": "Previous month",
33
+ icon: MiniChevronLeftIcon,
34
+ size: "small",
35
+ tip: "Previous month",
36
+ onClick: handlePreviousMonth
37
+ }), /*#__PURE__*/_jsx(Text, {
38
+ "aria-live": "polite",
39
+ as: "h2",
40
+ id: headingId,
41
+ variant: "title-sm",
42
+ children: monthYear
43
+ }), /*#__PURE__*/_jsx(IconButton, {
44
+ "aria-label": "Next month",
45
+ icon: MiniChevronRightIcon,
46
+ size: "small",
47
+ tip: "Next month",
48
+ onClick: handleNextMonth
49
+ })]
50
+ });
51
+ };
@@ -0,0 +1,6 @@
1
+ export { Calendar } from './Calendar';
2
+ export { CalendarHeader } from './CalendarHeader';
3
+ export { CalendarBody } from './CalendarBody';
4
+ export { CalendarFooter } from './CalendarFooter';
5
+ export type { CalendarHeaderProps, CalendarBodyProps, CalendarFooterProps, QuickAction, } from './types';
6
+ export * from './utils';
@@ -0,0 +1,5 @@
1
+ export { Calendar } from './Calendar';
2
+ export { CalendarHeader } from './CalendarHeader';
3
+ export { CalendarBody } from './CalendarBody';
4
+ export { CalendarFooter } from './CalendarFooter';
5
+ export * from './utils';
@@ -0,0 +1,57 @@
1
+ /**
2
+ * Internal types for the Calendar subcomponents (used by DatePickerCalendar).
3
+ */
4
+ export interface CalendarHeaderProps {
5
+ /** Currently displayed month and year (used for heading and prev/next range) */
6
+ currentMonthYear: Date;
7
+ /** Called when the user navigates to a different month. Pass the new date (e.g. setVisibleDate) so the calendar updates. */
8
+ onCurrentMonthYearChange: (newDate: Date) => void;
9
+ /** Optional. Called after navigating to previous month; use for click tracking. */
10
+ onPreviousMonthClick?: () => void;
11
+ /** Optional. Called after navigating to next month; use for click tracking. */
12
+ onNextMonthClick?: () => void;
13
+ /** Locale for month/year formatting (e.g. 'en-US') */
14
+ locale?: string;
15
+ /** Optional id for the heading (for grid aria-labelledby) */
16
+ headingId?: string;
17
+ }
18
+ export interface CalendarBodyProps {
19
+ /** The month to display (typically first day of that month) */
20
+ visibleDate: Date;
21
+ /** Selected start date (single or range start) */
22
+ selectedDate: Date | null;
23
+ /** Selected end date (range only; null for single-date mode) */
24
+ endDate?: Date | null;
25
+ /** Dates that should be disabled (unselectable) */
26
+ disabledDates?: Date[];
27
+ /** Called when a date cell is selected */
28
+ onDateSelect: (date: Date) => void;
29
+ /** Locale for weekday names and week start */
30
+ locale?: string;
31
+ /** 0 = Sunday, 1 = Monday (default from locale if not set) */
32
+ weekStartsOn?: 0 | 1;
33
+ /** Id of the month/year heading (aria-labelledby on grid) */
34
+ labelledById?: string;
35
+ /** For keyboard nav: which cell has focus (roving tabindex) */
36
+ focusedDate?: Date | null;
37
+ /** Callback when focused date changes (e.g. arrow keys) */
38
+ onFocusedDateChange?: (date: Date | null) => void;
39
+ /** Called when grid keyboard nav changes month (e.g. Page Up/Down). Pass setVisibleDate so the calendar updates. */
40
+ onVisibleDateChange?: (newDate: Date) => void;
41
+ /** Called when the escape key is pressed */
42
+ onEscapeKeyPress?: () => void;
43
+ }
44
+ export interface QuickAction {
45
+ num: number;
46
+ timePeriod: 'day' | 'week' | 'month' | 'year';
47
+ onClick: () => void;
48
+ }
49
+ export interface CalendarFooterProps {
50
+ onClearDate?: () => void;
51
+ onTodayClick?: () => void;
52
+ /** Called when the user navigates to a different month. Pass the new date (e.g. setVisibleDate) so the calendar updates. */
53
+ onSelectedDateChange: (newDate: Date | null) => void;
54
+ onCurrentMonthYearChange: (newDate: Date) => void;
55
+ /** Max 3 quick actions (e.g. "7 days", "1 month") */
56
+ quickActions?: QuickAction[];
57
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,34 @@
1
+ /**
2
+ * Builds a grid of days for a calendar month using native Date and Intl.
3
+ * Each row has 7 cells; leading/trailing cells may be null (padding from adjacent months).
4
+ */
5
+ /**
6
+ * Get the weekday for a date (0 = Sunday, 6 = Saturday).
7
+ * Optionally use weekStartsOn to compute "offset" for display (e.g. Monday = 0).
8
+ */
9
+ export declare function getDayOfWeek(date: Date, weekStartsOn?: 0 | 1): number;
10
+ /**
11
+ * Returns an array of weeks for the given month. Each week is an array of 7 items:
12
+ * each item is either a Date (that day) or null (padding from previous/next month).
13
+ *
14
+ * @param year - Full year (e.g. 2026)
15
+ * @param month - Month 0-11 (0 = January)
16
+ * @param weekStartsOn - 0 = Sunday, 1 = Monday
17
+ */
18
+ export declare function getMonthGrid(year: number, month: number, weekStartsOn?: 0 | 1): (Date | null)[][];
19
+ /**
20
+ * Check if two dates are the same calendar day (ignoring time).
21
+ */
22
+ export declare function isSameDay(a: Date | null, b: Date | null): boolean;
23
+ /**
24
+ * Check if `date` is between `start` and `end` (inclusive), ignoring time.
25
+ */
26
+ export declare function isDateInRange(date: Date, start: Date | null, end: Date | null): boolean;
27
+ /**
28
+ * Check if `date` is in the `disabledDates` list (by calendar day).
29
+ */
30
+ export declare function isDateDisabled(date: Date, disabledDates?: Date[]): boolean;
31
+ /**
32
+ * Clamp a day to the last day of the given month (e.g. Jan 31 -> Feb 28).
33
+ */
34
+ export declare function clampToMonth(year: number, month: number, day: number): Date;
@@ -0,0 +1,101 @@
1
+ /**
2
+ * Builds a grid of days for a calendar month using native Date and Intl.
3
+ * Each row has 7 cells; leading/trailing cells may be null (padding from adjacent months).
4
+ */
5
+
6
+ const DAYS_PER_WEEK = 7;
7
+
8
+ /**
9
+ * Normalize to start of day in local time for comparison.
10
+ */
11
+ function toDateOnly(d) {
12
+ return new Date(d.getFullYear(), d.getMonth(), d.getDate());
13
+ }
14
+
15
+ /**
16
+ * Get the weekday for a date (0 = Sunday, 6 = Saturday).
17
+ * Optionally use weekStartsOn to compute "offset" for display (e.g. Monday = 0).
18
+ */
19
+ export function getDayOfWeek(date, weekStartsOn = 0) {
20
+ const sundayBased = date.getDay();
21
+ if (weekStartsOn === 0) return sundayBased;
22
+ return (sundayBased + 6) % 7; // Monday = 0
23
+ }
24
+
25
+ /**
26
+ * Returns an array of weeks for the given month. Each week is an array of 7 items:
27
+ * each item is either a Date (that day) or null (padding from previous/next month).
28
+ *
29
+ * @param year - Full year (e.g. 2026)
30
+ * @param month - Month 0-11 (0 = January)
31
+ * @param weekStartsOn - 0 = Sunday, 1 = Monday
32
+ */
33
+ export function getMonthGrid(year, month, weekStartsOn = 0) {
34
+ const first = new Date(year, month, 1);
35
+ const last = new Date(year, month + 1, 0);
36
+ const firstDayOfWeek = getDayOfWeek(first, weekStartsOn);
37
+ const daysInMonth = last.getDate();
38
+ const weeks = [];
39
+ let currentWeek = [];
40
+
41
+ // Pad start of first week with nulls
42
+ // eslint-disable-next-line no-plusplus
43
+ for (let i = 0; i < firstDayOfWeek; i++) {
44
+ currentWeek.push(null);
45
+ }
46
+
47
+ // fix these
48
+ // eslint-disable-next-line no-plusplus
49
+ for (let day = 1; day <= daysInMonth; day++) {
50
+ currentWeek.push(new Date(year, month, day));
51
+ if (currentWeek.length === DAYS_PER_WEEK) {
52
+ weeks.push(currentWeek);
53
+ currentWeek = [];
54
+ }
55
+ }
56
+
57
+ // Pad end of last week with nulls
58
+ if (currentWeek.length > 0) {
59
+ while (currentWeek.length < DAYS_PER_WEEK) {
60
+ currentWeek.push(null);
61
+ }
62
+ weeks.push(currentWeek);
63
+ }
64
+ return weeks;
65
+ }
66
+
67
+ /**
68
+ * Check if two dates are the same calendar day (ignoring time).
69
+ */
70
+ export function isSameDay(a, b) {
71
+ if (a === null || b === null) return false;
72
+ return toDateOnly(a).getTime() === toDateOnly(b).getTime();
73
+ }
74
+
75
+ /**
76
+ * Check if `date` is between `start` and `end` (inclusive), ignoring time.
77
+ */
78
+ export function isDateInRange(date, start, end) {
79
+ if (start === null) return false;
80
+ const d = toDateOnly(date).getTime();
81
+ const s = toDateOnly(start).getTime();
82
+ const e = end !== null ? toDateOnly(end).getTime() : s;
83
+ const low = Math.min(s, e);
84
+ const high = Math.max(s, e);
85
+ return d >= low && d <= high;
86
+ }
87
+
88
+ /**
89
+ * Check if `date` is in the `disabledDates` list (by calendar day).
90
+ */
91
+ export function isDateDisabled(date, disabledDates = []) {
92
+ return disabledDates.some(d => isSameDay(date, d));
93
+ }
94
+
95
+ /**
96
+ * Clamp a day to the last day of the given month (e.g. Jan 31 -> Feb 28).
97
+ */
98
+ export function clampToMonth(year, month, day) {
99
+ const last = new Date(year, month + 1, 0).getDate();
100
+ return new Date(year, month, Math.min(day, last));
101
+ }
@@ -0,0 +1,39 @@
1
+ /**
2
+ * Date formatting for the calendar using Intl.DateTimeFormat.
3
+ */
4
+ /**
5
+ * Format month and year for the calendar header (e.g. "February 2026").
6
+ */
7
+ export declare function formatMonthYear(date: Date, locale?: string): string;
8
+ /**
9
+ * Get short weekday labels for column headers (e.g. ["Su", "Mo", ...]).
10
+ * Order depends on weekStartsOn: 0 = Sunday first, 1 = Monday first.
11
+ */
12
+ export declare function getWeekdayLabels(locale?: string, weekStartsOn?: 0 | 1): string[];
13
+ /**
14
+ * Get full weekday names for abbr attributes (e.g. "Sunday", "Monday").
15
+ * Same order as getWeekdayLabels.
16
+ */
17
+ export declare function getWeekdayFullNames(locale?: string, weekStartsOn?: 0 | 1): string[];
18
+ /**
19
+ * Format a date for display in the date picker input (e.g. "2/15/2026").
20
+ */
21
+ export declare function formatDateForInput(date: Date, locale?: string): string;
22
+ /**
23
+ * Parse a string from the date input into a Date, or null if invalid.
24
+ * Only returns a date when the input is a complete valid date (e.g. "2/15/2026").
25
+ * Partial input like "1" or "2/15" returns null even though Date("1") would parse.
26
+ */
27
+ export declare function parseDateFromInput(value: string, locale?: string): Date | null;
28
+ /**
29
+ * Format a date range for the input (e.g. "2/15/2026 – 2/20/2026").
30
+ */
31
+ export declare function formatDateRangeForInput(startDate: Date | null, endDate: Date | null, locale?: string): string;
32
+ /**
33
+ * Parse a range string (e.g. "2/15/2026 – 2/20/2026") into { startDate, endDate }.
34
+ * Returns null if invalid. Single date is allowed and yields startDate = endDate.
35
+ */
36
+ export declare function parseDateRangeFromInput(value: string, locale?: string): {
37
+ startDate: Date;
38
+ endDate: Date;
39
+ } | null;