@codecademy/gamut 68.1.3-alpha.ce69cb.0 → 68.1.3-alpha.da9068.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 (97) hide show
  1. package/dist/Alert/elements.d.ts +3 -3
  2. package/dist/Anchor/index.d.ts +9 -18
  3. package/dist/Anchor/index.js +6 -9
  4. package/dist/Box/GridBox.d.ts +0 -1
  5. package/dist/Box/GridBox.js +1 -1
  6. package/dist/Box/props.d.ts +1 -1
  7. package/dist/Button/CTAButton.d.ts +2 -2
  8. package/dist/Button/FillButton.d.ts +4 -4
  9. package/dist/Button/IconButton.d.ts +4 -4
  10. package/dist/Button/StrokeButton.d.ts +4 -4
  11. package/dist/Button/TextButton.d.ts +4 -4
  12. package/dist/Button/shared/InlineIconButton.d.ts +2 -2
  13. package/dist/Button/shared/styles.d.ts +3 -3
  14. package/dist/Button/shared/types.d.ts +1 -1
  15. package/dist/ButtonBase/ButtonBase.d.ts +4 -9
  16. package/dist/ButtonBase/ButtonBase.js +4 -11
  17. package/dist/Card/elements.d.ts +103 -109
  18. package/dist/Card/styles.d.ts +8 -8
  19. package/dist/Coachmark/index.d.ts +1 -1
  20. package/dist/ConnectedForm/utils.d.ts +1 -1
  21. package/dist/DatePicker/Calendar/Calendar.d.ts +9 -0
  22. package/dist/DatePicker/Calendar/Calendar.js +28 -0
  23. package/dist/DatePicker/Calendar/CalendarBody.d.ts +3 -0
  24. package/dist/DatePicker/Calendar/CalendarBody.js +174 -0
  25. package/dist/DatePicker/Calendar/CalendarFooter.d.ts +3 -0
  26. package/dist/DatePicker/Calendar/CalendarFooter.js +54 -0
  27. package/dist/DatePicker/Calendar/CalendarHeader.d.ts +3 -0
  28. package/dist/DatePicker/Calendar/CalendarHeader.js +86 -0
  29. package/dist/DatePicker/Calendar/index.d.ts +6 -0
  30. package/dist/DatePicker/Calendar/index.js +5 -0
  31. package/dist/DatePicker/Calendar/types.d.ts +64 -0
  32. package/dist/DatePicker/Calendar/types.js +1 -0
  33. package/dist/DatePicker/Calendar/utils/dateGrid.d.ts +30 -0
  34. package/dist/DatePicker/Calendar/utils/dateGrid.js +87 -0
  35. package/dist/DatePicker/Calendar/utils/format.d.ts +45 -0
  36. package/dist/DatePicker/Calendar/utils/format.js +123 -0
  37. package/dist/DatePicker/Calendar/utils/index.d.ts +3 -0
  38. package/dist/DatePicker/Calendar/utils/index.js +3 -0
  39. package/dist/DatePicker/Calendar/utils/keyHandler.d.ts +13 -0
  40. package/dist/DatePicker/Calendar/utils/keyHandler.js +125 -0
  41. package/dist/DatePicker/Calendar/utils/validation.d.ts +13 -0
  42. package/dist/DatePicker/Calendar/utils/validation.js +23 -0
  43. package/dist/DatePicker/DatePicker.d.ts +8 -0
  44. package/dist/DatePicker/DatePicker.js +128 -0
  45. package/dist/DatePicker/DatePickerCalendar.d.ts +13 -0
  46. package/dist/DatePicker/DatePickerCalendar.js +135 -0
  47. package/dist/DatePicker/DatePickerContext.d.ts +11 -0
  48. package/dist/DatePicker/DatePickerContext.js +18 -0
  49. package/dist/DatePicker/DatePickerInput.d.ts +16 -0
  50. package/dist/DatePicker/DatePickerInput.js +137 -0
  51. package/dist/DatePicker/index.d.ts +13 -0
  52. package/dist/DatePicker/index.js +10 -0
  53. package/dist/DatePicker/translations.d.ts +3 -0
  54. package/dist/DatePicker/translations.js +8 -0
  55. package/dist/DatePicker/types.d.ts +93 -0
  56. package/dist/DatePicker/types.js +1 -0
  57. package/dist/DatePicker/utils.d.ts +5 -0
  58. package/dist/DatePicker/utils.js +90 -0
  59. package/dist/Disclosure/elements.d.ts +12 -18
  60. package/dist/FeatureShimmer/index.js +1 -1
  61. package/dist/FocusTrap/index.d.ts +2 -2
  62. package/dist/Form/SelectDropdown/SelectDropdown.js +1 -1
  63. package/dist/Form/SelectDropdown/elements/controls.js +2 -2
  64. package/dist/Form/SelectDropdown/elements/multi-value.js +2 -2
  65. package/dist/Form/SelectDropdown/types/internal.d.ts +2 -2
  66. package/dist/Form/elements/Form.d.ts +15 -15
  67. package/dist/Form/elements/FormGroup.d.ts +1 -1
  68. package/dist/GridForm/GridFormButtons/index.d.ts +4 -4
  69. package/dist/List/ListProvider.d.ts +1 -1
  70. package/dist/List/elements.d.ts +43 -45
  71. package/dist/Menu/MenuItem.js +6 -10
  72. package/dist/Menu/elements.d.ts +2 -2
  73. package/dist/Modals/elements.d.ts +1 -1
  74. package/dist/Pagination/AnimatedPaginationButtons.d.ts +32 -34
  75. package/dist/Pagination/EllipsisButton.d.ts +4 -4
  76. package/dist/Pagination/PaginationButton.d.ts +6 -6
  77. package/dist/Pagination/utils.d.ts +30 -32
  78. package/dist/Pagination/utils.js +11 -14
  79. package/dist/Popover/Popover.js +4 -4
  80. package/dist/Popover/types.d.ts +2 -3
  81. package/dist/PopoverContainer/PopoverContainer.js +13 -9
  82. package/dist/PopoverContainer/hooks.d.ts +4 -16
  83. package/dist/PopoverContainer/hooks.js +24 -31
  84. package/dist/PopoverContainer/types.d.ts +7 -3
  85. package/dist/Tabs/TabButton.d.ts +2 -2
  86. package/dist/Tabs/TabNavLink.d.ts +2 -2
  87. package/dist/Tag/elements.d.ts +8 -14
  88. package/dist/Tip/InfoTip/InfoTipButton.d.ts +4 -4
  89. package/dist/Tip/PreviewTip/elements.d.ts +6 -12
  90. package/dist/Tip/__tests__/helpers.d.ts +1 -1
  91. package/dist/Tip/shared/FloatingTip.js +2 -2
  92. package/dist/Tip/shared/types.d.ts +2 -2
  93. package/dist/Typography/Text.d.ts +1 -1
  94. package/dist/index.d.ts +1 -0
  95. package/dist/index.js +1 -0
  96. package/dist/utils/react.js +1 -2
  97. package/package.json +11 -11
@@ -9,14 +9,14 @@ export declare const patternFadeInOut: {
9
9
  opacity: number;
10
10
  transition: {
11
11
  duration: number;
12
- ease: "easeOut";
12
+ ease: string;
13
13
  };
14
14
  };
15
15
  animate: {
16
16
  opacity: number;
17
17
  transition: {
18
18
  duration: number;
19
- ease: "easeIn";
19
+ ease: string;
20
20
  };
21
21
  };
22
22
  };
@@ -26,7 +26,7 @@ export declare const hoverShadowLeft: (borderRadius?: string) => {
26
26
  borderRadius: string | undefined;
27
27
  transition: {
28
28
  duration: number;
29
- ease: "easeOut";
29
+ ease: string;
30
30
  };
31
31
  };
32
32
  initialOutline: {
@@ -34,7 +34,7 @@ export declare const hoverShadowLeft: (borderRadius?: string) => {
34
34
  borderRadius: string | undefined;
35
35
  transition: {
36
36
  duration: number;
37
- ease: "easeOut";
37
+ ease: string;
38
38
  };
39
39
  };
40
40
  animate: {
@@ -43,7 +43,7 @@ export declare const hoverShadowLeft: (borderRadius?: string) => {
43
43
  borderRadius: string | undefined;
44
44
  transition: {
45
45
  duration: number;
46
- ease: "easeIn";
46
+ ease: string;
47
47
  };
48
48
  };
49
49
  animateOutline: {
@@ -52,7 +52,7 @@ export declare const hoverShadowLeft: (borderRadius?: string) => {
52
52
  borderRadius: string | undefined;
53
53
  transition: {
54
54
  duration: number;
55
- ease: "easeIn";
55
+ ease: string;
56
56
  };
57
57
  };
58
58
  };
@@ -62,7 +62,7 @@ export declare const hoverShadowRight: (borderRadius?: string) => {
62
62
  borderRadius: string | undefined;
63
63
  transition: {
64
64
  duration: number;
65
- ease: "easeOut";
65
+ ease: string;
66
66
  };
67
67
  };
68
68
  animate: {
@@ -71,7 +71,7 @@ export declare const hoverShadowRight: (borderRadius?: string) => {
71
71
  borderRadius: string | undefined;
72
72
  transition: {
73
73
  duration: number;
74
- ease: "easeIn";
74
+ ease: string;
75
75
  };
76
76
  };
77
77
  };
@@ -21,7 +21,7 @@ export type CoachmarkProps = PopoverFocusProps & {
21
21
  /**
22
22
  * Function that returns the contents of the coachmark.
23
23
  */
24
- renderPopover: (onDismiss?: () => void) => React.JSX.Element;
24
+ renderPopover: (onDismiss?: () => void) => JSX.Element;
25
25
  /**
26
26
  * Props to be passed into the popover component.
27
27
  */
@@ -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,9 @@
1
+ import * as React from 'react';
2
+ /**
3
+ * Outer wrapper for the calendar (header + body + footer).
4
+ * Used by DatePickerCalendar to group the calendar content.
5
+ * Renders a CheckerDense pattern shadow at offset left 8, top 8.
6
+ */
7
+ export declare const Calendar: React.FC<{
8
+ children: React.ReactNode;
9
+ }>;
@@ -0,0 +1,28 @@
1
+ import { CheckerDense } from '@codecademy/gamut-patterns';
2
+ import * as React from 'react';
3
+ import { Box } from '../../Box';
4
+
5
+ /**
6
+ * Outer wrapper for the calendar (header + body + footer).
7
+ * Used by DatePickerCalendar to group the calendar content.
8
+ * Renders a CheckerDense pattern shadow at offset left 8, top 8.
9
+ */
10
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
11
+ export const Calendar = ({
12
+ children
13
+ }) => /*#__PURE__*/_jsxs(Box, {
14
+ position: "relative",
15
+ width: "max-content",
16
+ children: [/*#__PURE__*/_jsx(CheckerDense, {
17
+ left: 8,
18
+ position: "absolute",
19
+ top: 8
20
+ }), /*#__PURE__*/_jsx(Box, {
21
+ bg: "background",
22
+ border: 1,
23
+ borderRadius: "sm",
24
+ position: "relative",
25
+ zIndex: 1,
26
+ children: children
27
+ })]
28
+ });
@@ -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,174 @@
1
+ import _styled from "@emotion/styled/base";
2
+ import { css, 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 { getMonthGrid, isDateDisabled, isDateInRange, isSameDay } from './utils/dateGrid';
7
+ import { getWeekdayNames } from './utils/format';
8
+ import { getDatesWithRow, keyHandler } from './utils/keyHandler';
9
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
10
+ const TableHeader = /*#__PURE__*/_styled("th", {
11
+ target: "e12sl4cx2",
12
+ label: "TableHeader"
13
+ })(css({
14
+ fontSize: 14,
15
+ fontWeight: 'base',
16
+ color: 'text-disabled',
17
+ textAlign: 'center'
18
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DatePicker/Calendar/CalendarBody.tsx"],"names":[],"mappings":"AAgBoB","file":"../../../src/DatePicker/Calendar/CalendarBody.tsx","sourcesContent":["import { css, 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  getMonthGrid,\n  isDateDisabled,\n  isDateInRange,\n  isSameDay,\n} from './utils/dateGrid';\nimport { getWeekdayNames } from './utils/format';\nimport { getDatesWithRow, keyHandler } from './utils/keyHandler';\n\nconst TableHeader = styled.th(\n  css({\n    fontSize: 14,\n    fontWeight: 'base',\n    color: 'text-disabled',\n    textAlign: 'center',\n  })\n);\n\nconst DateCell = styled.td(\n  css({\n    padding: 0,\n  })\n);\n\nconst DateButton = styled(TextButton)(\n  states({\n    isToday: {\n      position: 'relative',\n      '&::after': {\n        content: '\"\"',\n        position: 'absolute',\n        bottom: 4,\n        width: 4,\n        height: 4,\n        borderRadius: 'full',\n        bg: 'hyper',\n      },\n    },\n    isSelected: {\n      bg: 'text',\n      color: 'background',\n      '&:hover, &:focus': {\n        bg: 'secondary-hover',\n        color: 'background',\n      },\n      '&::after': {\n        bg: 'background',\n      },\n    },\n    isRangeStart: {\n      borderRadiusRight: 'none',\n    },\n    isRangeEnd: {\n      borderRadiusLeft: 'none',\n    },\n    isInRange: {\n      bg: 'text-disabled',\n      color: 'background',\n      borderRadius: 'none',\n      '&:hover, &:focus': {\n        bg: 'secondary-hover',\n        color: 'background',\n      },\n      '&::after': {\n        bg: 'background',\n      },\n    },\n    disabled: {\n      color: 'text-disabled',\n      textDecoration: 'line-through',\n      '&:hover': {\n        textDecoration: 'line-through',\n      },\n    },\n  }),\n  css({\n    fontWeight: 'base',\n    width: '32px',\n  })\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  hasAdjacentMonthRight,\n  hasAdjacentMonthLeft,\n}) => {\n  const year = visibleDate.getFullYear();\n  const month = visibleDate.getMonth();\n  const weeks = getMonthGrid(year, month, weekStartsOn);\n  const weekdayLabels = getWeekdayNames('short', locale, weekStartsOn);\n  const weekdayFullNames = getWeekdayNames('long', 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    (date: Date | null) => date !== null && isSameDay(date, 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      keyHandler(\n        e,\n        date,\n        onFocusedDateChange,\n        datesWithRow,\n        month,\n        year,\n        disabledDates,\n        onDateSelect,\n        onEscapeKeyPress,\n        onVisibleDateChange,\n        hasAdjacentMonthRight,\n        hasAdjacentMonthLeft\n      ),\n    [\n      onFocusedDateChange,\n      datesWithRow,\n      month,\n      year,\n      disabledDates,\n      onDateSelect,\n      onEscapeKeyPress,\n      onVisibleDateChange,\n      hasAdjacentMonthLeft,\n      hasAdjacentMonthRight,\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            <TableHeader abbr={weekdayFullNames[i]} key={label} scope=\"col\">\n              {label}\n            </TableHeader>\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                  <DateCell\n                    key={`empty-${rowIndex}-${colIndex}`}\n                    role=\"gridcell\"\n                  />\n                );\n              }\n              const selected =\n                isSameDay(date, selectedDate) || isSameDay(date, endDate);\n              const range = !!selectedDate && !!endDate;\n              const inRange =\n                range && isDateInRange(date, selectedDate, endDate);\n              const disabled = isDateDisabled(date, disabledDates);\n              const today = isToday(date);\n              // this is making the selected date a differnet color bc it is focused, look into further\n              const isFocused =\n                focusTarget !== null && isSameDay(date, focusTarget);\n\n              return (\n                <DateCell\n                  aria-selected={selected}\n                  key={date.getTime()}\n                  role=\"gridcell\"\n                >\n                  <DateButton\n                    disabled={disabled}\n                    isInRange={inRange}\n                    isRangeEnd={range && isSameDay(date, endDate)}\n                    isRangeStart={range && isSameDay(date, selectedDate)}\n                    isSelected={selected}\n                    isToday={today}\n                    ref={(el) => setButtonRef(date, el as HTMLElement | null)}\n                    tabIndex={isFocused ? 0 : -1}\n                    variant=\"secondary\"\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                </DateCell>\n              );\n            })}\n          </tr>\n        ))}\n      </tbody>\n    </table>\n  );\n};\n"]} */");
19
+ const DateCell = /*#__PURE__*/_styled("td", {
20
+ target: "e12sl4cx1",
21
+ label: "DateCell"
22
+ })(css({
23
+ padding: 0
24
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DatePicker/Calendar/CalendarBody.tsx"],"names":[],"mappings":"AAyBiB","file":"../../../src/DatePicker/Calendar/CalendarBody.tsx","sourcesContent":["import { css, 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  getMonthGrid,\n  isDateDisabled,\n  isDateInRange,\n  isSameDay,\n} from './utils/dateGrid';\nimport { getWeekdayNames } from './utils/format';\nimport { getDatesWithRow, keyHandler } from './utils/keyHandler';\n\nconst TableHeader = styled.th(\n  css({\n    fontSize: 14,\n    fontWeight: 'base',\n    color: 'text-disabled',\n    textAlign: 'center',\n  })\n);\n\nconst DateCell = styled.td(\n  css({\n    padding: 0,\n  })\n);\n\nconst DateButton = styled(TextButton)(\n  states({\n    isToday: {\n      position: 'relative',\n      '&::after': {\n        content: '\"\"',\n        position: 'absolute',\n        bottom: 4,\n        width: 4,\n        height: 4,\n        borderRadius: 'full',\n        bg: 'hyper',\n      },\n    },\n    isSelected: {\n      bg: 'text',\n      color: 'background',\n      '&:hover, &:focus': {\n        bg: 'secondary-hover',\n        color: 'background',\n      },\n      '&::after': {\n        bg: 'background',\n      },\n    },\n    isRangeStart: {\n      borderRadiusRight: 'none',\n    },\n    isRangeEnd: {\n      borderRadiusLeft: 'none',\n    },\n    isInRange: {\n      bg: 'text-disabled',\n      color: 'background',\n      borderRadius: 'none',\n      '&:hover, &:focus': {\n        bg: 'secondary-hover',\n        color: 'background',\n      },\n      '&::after': {\n        bg: 'background',\n      },\n    },\n    disabled: {\n      color: 'text-disabled',\n      textDecoration: 'line-through',\n      '&:hover': {\n        textDecoration: 'line-through',\n      },\n    },\n  }),\n  css({\n    fontWeight: 'base',\n    width: '32px',\n  })\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  hasAdjacentMonthRight,\n  hasAdjacentMonthLeft,\n}) => {\n  const year = visibleDate.getFullYear();\n  const month = visibleDate.getMonth();\n  const weeks = getMonthGrid(year, month, weekStartsOn);\n  const weekdayLabels = getWeekdayNames('short', locale, weekStartsOn);\n  const weekdayFullNames = getWeekdayNames('long', 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    (date: Date | null) => date !== null && isSameDay(date, 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      keyHandler(\n        e,\n        date,\n        onFocusedDateChange,\n        datesWithRow,\n        month,\n        year,\n        disabledDates,\n        onDateSelect,\n        onEscapeKeyPress,\n        onVisibleDateChange,\n        hasAdjacentMonthRight,\n        hasAdjacentMonthLeft\n      ),\n    [\n      onFocusedDateChange,\n      datesWithRow,\n      month,\n      year,\n      disabledDates,\n      onDateSelect,\n      onEscapeKeyPress,\n      onVisibleDateChange,\n      hasAdjacentMonthLeft,\n      hasAdjacentMonthRight,\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            <TableHeader abbr={weekdayFullNames[i]} key={label} scope=\"col\">\n              {label}\n            </TableHeader>\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                  <DateCell\n                    key={`empty-${rowIndex}-${colIndex}`}\n                    role=\"gridcell\"\n                  />\n                );\n              }\n              const selected =\n                isSameDay(date, selectedDate) || isSameDay(date, endDate);\n              const range = !!selectedDate && !!endDate;\n              const inRange =\n                range && isDateInRange(date, selectedDate, endDate);\n              const disabled = isDateDisabled(date, disabledDates);\n              const today = isToday(date);\n              // this is making the selected date a differnet color bc it is focused, look into further\n              const isFocused =\n                focusTarget !== null && isSameDay(date, focusTarget);\n\n              return (\n                <DateCell\n                  aria-selected={selected}\n                  key={date.getTime()}\n                  role=\"gridcell\"\n                >\n                  <DateButton\n                    disabled={disabled}\n                    isInRange={inRange}\n                    isRangeEnd={range && isSameDay(date, endDate)}\n                    isRangeStart={range && isSameDay(date, selectedDate)}\n                    isSelected={selected}\n                    isToday={today}\n                    ref={(el) => setButtonRef(date, el as HTMLElement | null)}\n                    tabIndex={isFocused ? 0 : -1}\n                    variant=\"secondary\"\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                </DateCell>\n              );\n            })}\n          </tr>\n        ))}\n      </tbody>\n    </table>\n  );\n};\n"]} */");
25
+ const DateButton = /*#__PURE__*/_styled(TextButton, {
26
+ target: "e12sl4cx0",
27
+ label: "DateButton"
28
+ })(states({
29
+ isToday: {
30
+ position: 'relative',
31
+ '&::after': {
32
+ content: '""',
33
+ position: 'absolute',
34
+ bottom: 4,
35
+ width: 4,
36
+ height: 4,
37
+ borderRadius: 'full',
38
+ bg: 'hyper'
39
+ }
40
+ },
41
+ isSelected: {
42
+ bg: 'text',
43
+ color: 'background',
44
+ '&:hover, &:focus': {
45
+ bg: 'secondary-hover',
46
+ color: 'background'
47
+ },
48
+ '&::after': {
49
+ bg: 'background'
50
+ }
51
+ },
52
+ isRangeStart: {
53
+ borderRadiusRight: 'none'
54
+ },
55
+ isRangeEnd: {
56
+ borderRadiusLeft: 'none'
57
+ },
58
+ isInRange: {
59
+ bg: 'text-disabled',
60
+ color: 'background',
61
+ borderRadius: 'none',
62
+ '&:hover, &:focus': {
63
+ bg: 'secondary-hover',
64
+ color: 'background'
65
+ },
66
+ '&::after': {
67
+ bg: 'background'
68
+ }
69
+ },
70
+ disabled: {
71
+ color: 'text-disabled',
72
+ textDecoration: 'line-through',
73
+ '&:hover': {
74
+ textDecoration: 'line-through'
75
+ }
76
+ }
77
+ }), css({
78
+ fontWeight: 'base',
79
+ width: '32px'
80
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DatePicker/Calendar/CalendarBody.tsx"],"names":[],"mappings":"AA+BmB","file":"../../../src/DatePicker/Calendar/CalendarBody.tsx","sourcesContent":["import { css, 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  getMonthGrid,\n  isDateDisabled,\n  isDateInRange,\n  isSameDay,\n} from './utils/dateGrid';\nimport { getWeekdayNames } from './utils/format';\nimport { getDatesWithRow, keyHandler } from './utils/keyHandler';\n\nconst TableHeader = styled.th(\n  css({\n    fontSize: 14,\n    fontWeight: 'base',\n    color: 'text-disabled',\n    textAlign: 'center',\n  })\n);\n\nconst DateCell = styled.td(\n  css({\n    padding: 0,\n  })\n);\n\nconst DateButton = styled(TextButton)(\n  states({\n    isToday: {\n      position: 'relative',\n      '&::after': {\n        content: '\"\"',\n        position: 'absolute',\n        bottom: 4,\n        width: 4,\n        height: 4,\n        borderRadius: 'full',\n        bg: 'hyper',\n      },\n    },\n    isSelected: {\n      bg: 'text',\n      color: 'background',\n      '&:hover, &:focus': {\n        bg: 'secondary-hover',\n        color: 'background',\n      },\n      '&::after': {\n        bg: 'background',\n      },\n    },\n    isRangeStart: {\n      borderRadiusRight: 'none',\n    },\n    isRangeEnd: {\n      borderRadiusLeft: 'none',\n    },\n    isInRange: {\n      bg: 'text-disabled',\n      color: 'background',\n      borderRadius: 'none',\n      '&:hover, &:focus': {\n        bg: 'secondary-hover',\n        color: 'background',\n      },\n      '&::after': {\n        bg: 'background',\n      },\n    },\n    disabled: {\n      color: 'text-disabled',\n      textDecoration: 'line-through',\n      '&:hover': {\n        textDecoration: 'line-through',\n      },\n    },\n  }),\n  css({\n    fontWeight: 'base',\n    width: '32px',\n  })\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  hasAdjacentMonthRight,\n  hasAdjacentMonthLeft,\n}) => {\n  const year = visibleDate.getFullYear();\n  const month = visibleDate.getMonth();\n  const weeks = getMonthGrid(year, month, weekStartsOn);\n  const weekdayLabels = getWeekdayNames('short', locale, weekStartsOn);\n  const weekdayFullNames = getWeekdayNames('long', 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    (date: Date | null) => date !== null && isSameDay(date, 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      keyHandler(\n        e,\n        date,\n        onFocusedDateChange,\n        datesWithRow,\n        month,\n        year,\n        disabledDates,\n        onDateSelect,\n        onEscapeKeyPress,\n        onVisibleDateChange,\n        hasAdjacentMonthRight,\n        hasAdjacentMonthLeft\n      ),\n    [\n      onFocusedDateChange,\n      datesWithRow,\n      month,\n      year,\n      disabledDates,\n      onDateSelect,\n      onEscapeKeyPress,\n      onVisibleDateChange,\n      hasAdjacentMonthLeft,\n      hasAdjacentMonthRight,\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            <TableHeader abbr={weekdayFullNames[i]} key={label} scope=\"col\">\n              {label}\n            </TableHeader>\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                  <DateCell\n                    key={`empty-${rowIndex}-${colIndex}`}\n                    role=\"gridcell\"\n                  />\n                );\n              }\n              const selected =\n                isSameDay(date, selectedDate) || isSameDay(date, endDate);\n              const range = !!selectedDate && !!endDate;\n              const inRange =\n                range && isDateInRange(date, selectedDate, endDate);\n              const disabled = isDateDisabled(date, disabledDates);\n              const today = isToday(date);\n              // this is making the selected date a differnet color bc it is focused, look into further\n              const isFocused =\n                focusTarget !== null && isSameDay(date, focusTarget);\n\n              return (\n                <DateCell\n                  aria-selected={selected}\n                  key={date.getTime()}\n                  role=\"gridcell\"\n                >\n                  <DateButton\n                    disabled={disabled}\n                    isInRange={inRange}\n                    isRangeEnd={range && isSameDay(date, endDate)}\n                    isRangeStart={range && isSameDay(date, selectedDate)}\n                    isSelected={selected}\n                    isToday={today}\n                    ref={(el) => setButtonRef(date, el as HTMLElement | null)}\n                    tabIndex={isFocused ? 0 : -1}\n                    variant=\"secondary\"\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                </DateCell>\n              );\n            })}\n          </tr>\n        ))}\n      </tbody>\n    </table>\n  );\n};\n"]} */");
81
+ export const CalendarBody = ({
82
+ visibleDate,
83
+ selectedDate,
84
+ endDate = null,
85
+ disabledDates = [],
86
+ onDateSelect,
87
+ locale,
88
+ weekStartsOn = 0,
89
+ labelledById,
90
+ focusedDate,
91
+ onFocusedDateChange,
92
+ onVisibleDateChange,
93
+ onEscapeKeyPress,
94
+ hasAdjacentMonthRight,
95
+ hasAdjacentMonthLeft
96
+ }) => {
97
+ const year = visibleDate.getFullYear();
98
+ const month = visibleDate.getMonth();
99
+ const weeks = getMonthGrid(year, month, weekStartsOn);
100
+ const weekdayLabels = getWeekdayNames('short', locale, weekStartsOn);
101
+ const weekdayFullNames = getWeekdayNames('long', locale, weekStartsOn);
102
+ const buttonRefs = useRef(new Map());
103
+ const datesWithRow = useMemo(() => getDatesWithRow(weeks), [weeks]);
104
+ const focusTarget = focusedDate ?? selectedDate;
105
+ const isToday = useCallback(date => date !== null && isSameDay(date, new Date()), []);
106
+ const focusButton = useCallback(date => {
107
+ if (date === null) return;
108
+ const key = new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime();
109
+ buttonRefs.current.get(key)?.focus();
110
+ }, []);
111
+ useEffect(() => {
112
+ if (focusTarget !== null) focusButton(focusTarget);
113
+ }, [focusTarget, focusButton]);
114
+ const handleKeyDown = useCallback((e, date) => keyHandler(e, date, onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onVisibleDateChange, hasAdjacentMonthRight, hasAdjacentMonthLeft), [onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onVisibleDateChange, hasAdjacentMonthLeft, hasAdjacentMonthRight]);
115
+ const setButtonRef = useCallback((date, el) => {
116
+ const k = new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime();
117
+ if (el) buttonRefs.current.set(k, el);else buttonRefs.current.delete(k);
118
+ }, []);
119
+ return /*#__PURE__*/_jsxs("table", {
120
+ "aria-labelledby": labelledById,
121
+ role: "grid",
122
+ width: "100%",
123
+ children: [/*#__PURE__*/_jsx("thead", {
124
+ children: /*#__PURE__*/_jsx("tr", {
125
+ children: weekdayLabels.map((label, i) => /*#__PURE__*/_jsx(TableHeader, {
126
+ abbr: weekdayFullNames[i],
127
+ scope: "col",
128
+ children: label
129
+ }, label))
130
+ })
131
+ }), /*#__PURE__*/_jsx("tbody", {
132
+ children: weeks.map((week, rowIndex) => /*#__PURE__*/_jsx("tr", {
133
+ children: week.map((date, colIndex) => {
134
+ if (date === null) {
135
+ return (
136
+ /*#__PURE__*/
137
+ // fix this error
138
+ // eslint-disable-next-line react/no-array-index-key, jsx-a11y/control-has-associated-label
139
+ _jsx(DateCell, {
140
+ role: "gridcell"
141
+ }, `empty-${rowIndex}-${colIndex}`)
142
+ );
143
+ }
144
+ const selected = isSameDay(date, selectedDate) || isSameDay(date, endDate);
145
+ const range = !!selectedDate && !!endDate;
146
+ const inRange = range && isDateInRange(date, selectedDate, endDate);
147
+ const disabled = isDateDisabled(date, disabledDates);
148
+ const today = isToday(date);
149
+ // this is making the selected date a differnet color bc it is focused, look into further
150
+ const isFocused = focusTarget !== null && isSameDay(date, focusTarget);
151
+ return /*#__PURE__*/_jsx(DateCell, {
152
+ "aria-selected": selected,
153
+ role: "gridcell",
154
+ children: /*#__PURE__*/_jsx(DateButton, {
155
+ disabled: disabled,
156
+ isInRange: inRange,
157
+ isRangeEnd: range && isSameDay(date, endDate),
158
+ isRangeStart: range && isSameDay(date, selectedDate),
159
+ isSelected: selected,
160
+ isToday: today,
161
+ ref: el => setButtonRef(date, el),
162
+ tabIndex: isFocused ? 0 : -1,
163
+ variant: "secondary",
164
+ onClick: () => onDateSelect(date),
165
+ onFocus: () => onFocusedDateChange?.(date),
166
+ onKeyDown: e => handleKeyDown(e, date),
167
+ children: date.getDate()
168
+ })
169
+ }, date.getTime());
170
+ })
171
+ }, week.join('-')))
172
+ })]
173
+ });
174
+ };
@@ -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,54 @@
1
+ import * as React from 'react';
2
+ import { FlexBox } from '../../Box';
3
+ import { TextButton } from '../../Button';
4
+ import { getRelativeTodayLabel } from './utils/format';
5
+
6
+ // function formatQuickActionLabel(action: QuickAction): string {
7
+ // const { num, timePeriod } = action;
8
+ // const period =
9
+ // timePeriod === 'day'
10
+ // ? num === 1
11
+ // ? 'day'
12
+ // : 'days'
13
+ // : timePeriod === 'week'
14
+ // ? num === 1
15
+ // ? 'week'
16
+ // : 'weeks'
17
+ // : timePeriod === 'month'
18
+ // ? num === 1
19
+ // ? 'month'
20
+ // : 'months'
21
+ // : num === 1
22
+ // ? 'year'
23
+ // : 'years';
24
+ // return `${num} ${period}`;
25
+ // }
26
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
27
+ export const CalendarFooter = ({
28
+ onClearDate,
29
+ onTodayClick,
30
+ locale,
31
+ clearText,
32
+ disabled,
33
+ showClearButton
34
+ }) => {
35
+ // const actions = quickActions.slice(0, 3);
36
+
37
+ return /*#__PURE__*/_jsxs(FlexBox, {
38
+ alignItems: "center",
39
+ borderTop: 1,
40
+ justifyContent: "space-between",
41
+ p: 12,
42
+ children: [showClearButton && /*#__PURE__*/_jsx(TextButton, {
43
+ disabled: disabled,
44
+ onClick: () => onClearDate?.(),
45
+ children: clearText
46
+ }), /*#__PURE__*/_jsx(FlexBox, {
47
+ gap: 32,
48
+ children: /*#__PURE__*/_jsx(TextButton, {
49
+ onClick: () => onTodayClick?.(),
50
+ children: getRelativeTodayLabel(locale)
51
+ })
52
+ })]
53
+ });
54
+ };
@@ -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,86 @@
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, getRelativeMonthLabels } from './utils/format';
7
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
8
+ export const CalendarHeader = ({
9
+ currentMonthYear,
10
+ onCurrentMonthYearChange,
11
+ secondMonthYear,
12
+ onLastMonthClick,
13
+ onNextMonthClick,
14
+ locale,
15
+ headingId
16
+ }) => {
17
+ const {
18
+ nextMonth,
19
+ lastMonth
20
+ } = getRelativeMonthLabels(locale);
21
+ const handleLastMonth = () => {
22
+ const lastMonth = new Date(currentMonthYear.getFullYear(), currentMonthYear.getMonth() - 1, 1);
23
+ onCurrentMonthYearChange?.(lastMonth);
24
+ onLastMonthClick?.();
25
+ };
26
+ const handleNextMonth = () => {
27
+ const nextMonth = new Date(currentMonthYear.getFullYear(), currentMonthYear.getMonth() + 1, 1);
28
+ onCurrentMonthYearChange?.(nextMonth);
29
+ onNextMonthClick?.();
30
+ };
31
+ return /*#__PURE__*/_jsxs(FlexBox, {
32
+ alignItems: "center",
33
+ pb: 16,
34
+ width: "100%",
35
+ children: [/*#__PURE__*/_jsx(IconButton, {
36
+ "aria-label": lastMonth,
37
+ icon: MiniChevronLeftIcon,
38
+ size: "small",
39
+ tip: lastMonth,
40
+ onClick: handleLastMonth
41
+ }), /*#__PURE__*/_jsxs(FlexBox, {
42
+ flexGrow: 1,
43
+ gap: {
44
+ _: 0,
45
+ xs: secondMonthYear ? 32 : 0
46
+ },
47
+ minWidth: 0,
48
+ children: [/*#__PURE__*/_jsx(FlexBox, {
49
+ flexGrow: 1,
50
+ justifyContent: "center",
51
+ minWidth: 0,
52
+ children: /*#__PURE__*/_jsx(Text, {
53
+ "aria-live": "polite",
54
+ as: "h2",
55
+ fontSize: 16,
56
+ fontWeight: "title",
57
+ id: headingId,
58
+ textAlign: "center",
59
+ children: formatMonthYear(currentMonthYear, locale)
60
+ })
61
+ }), secondMonthYear && /*#__PURE__*/_jsx(FlexBox, {
62
+ display: {
63
+ _: 'none',
64
+ xs: 'flex'
65
+ },
66
+ flexGrow: 1,
67
+ justifyContent: "center",
68
+ minWidth: 0,
69
+ children: /*#__PURE__*/_jsx(Text, {
70
+ "aria-live": "polite",
71
+ as: "h2",
72
+ fontSize: 16,
73
+ fontWeight: "title",
74
+ textAlign: "center",
75
+ children: formatMonthYear(secondMonthYear, locale)
76
+ })
77
+ })]
78
+ }), /*#__PURE__*/_jsx(IconButton, {
79
+ "aria-label": nextMonth,
80
+ icon: MiniChevronRightIcon,
81
+ size: "small",
82
+ tip: nextMonth,
83
+ onClick: handleNextMonth
84
+ })]
85
+ });
86
+ };
@@ -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,64 @@
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
+ /** Currently displayed second month and year (used for heading and prev/next range) */
10
+ secondMonthYear?: Date;
11
+ /** Called after navigating to previous month; use for click tracking. */
12
+ onLastMonthClick?: () => void;
13
+ /** Called after navigating to next month; use for click tracking. */
14
+ onNextMonthClick?: () => void;
15
+ /** Locale for month/year formatting (e.g. 'en-US') */
16
+ locale?: string;
17
+ /** id for the heading (for grid aria-labelledby) */
18
+ headingId: string;
19
+ }
20
+ export interface CalendarBodyProps {
21
+ /** The month to display (typically first day of that month) */
22
+ visibleDate: Date;
23
+ /** Selected start date (single or range start) */
24
+ selectedDate: Date | null;
25
+ /** Selected end date (range only; undefined for single-date mode) */
26
+ endDate?: Date | null;
27
+ /** Dates that should be disabled (unselectable) */
28
+ disabledDates?: Date[];
29
+ /** Called when a date cell is selected */
30
+ onDateSelect: (date: Date) => void;
31
+ /** Locale for weekday names and week start */
32
+ locale?: string;
33
+ /** 0 = Sunday, 1 = Monday (default from locale if not set) */
34
+ weekStartsOn?: 0 | 1;
35
+ /** Id of the month/year heading (aria-labelledby on grid) */
36
+ labelledById: string;
37
+ /** For keyboard nav: which cell has focus (roving tabindex) */
38
+ focusedDate: Date | null;
39
+ /** Callback when focused date changes (e.g. arrow keys) */
40
+ onFocusedDateChange: (date: Date | null) => void;
41
+ /** Called when grid keyboard nav changes month (e.g. Page Up/Down). Pass setVisibleDate so the calendar updates. */
42
+ onVisibleDateChange: (newDate: Date) => void;
43
+ /** Called when the escape key is pressed */
44
+ onEscapeKeyPress?: () => void;
45
+ /** When true (e.g. two-month view), arrow keys move focus to adjacent month without changing visible date. */
46
+ hasAdjacentMonthRight?: boolean;
47
+ /** When true (e.g. two-month view), arrow keys move focus to adjacent month without changing visible date. */
48
+ hasAdjacentMonthLeft?: boolean;
49
+ }
50
+ export interface QuickAction {
51
+ num: number;
52
+ timePeriod: 'day' | 'week' | 'month' | 'year';
53
+ onClick: () => void;
54
+ }
55
+ export interface CalendarFooterProps {
56
+ disabled?: boolean;
57
+ showClearButton?: boolean;
58
+ locale?: string;
59
+ clearText: string;
60
+ onClearDate?: () => void;
61
+ onTodayClick?: () => void;
62
+ /** Max 3 quick actions (e.g. "7 days", "1 month") */
63
+ quickActions?: QuickAction[];
64
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,30 @@
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 const 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 const 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 const isSameDay: (a: Date | null, b: Date | null) => boolean;
23
+ /**
24
+ * Check if `date` is between `start` and `end` (exclusive), ignoring time.
25
+ */
26
+ export declare const 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 const isDateDisabled: (date: Date, disabledDates?: Date[]) => boolean;