@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.
- package/dist/Alert/elements.d.ts +3 -3
- package/dist/Anchor/index.d.ts +9 -18
- package/dist/Anchor/index.js +6 -9
- package/dist/Box/GridBox.d.ts +0 -1
- package/dist/Box/GridBox.js +1 -1
- package/dist/Box/props.d.ts +1 -1
- package/dist/Button/CTAButton.d.ts +2 -2
- package/dist/Button/FillButton.d.ts +4 -4
- package/dist/Button/IconButton.d.ts +4 -4
- package/dist/Button/StrokeButton.d.ts +4 -4
- package/dist/Button/TextButton.d.ts +4 -4
- package/dist/Button/shared/InlineIconButton.d.ts +2 -2
- package/dist/Button/shared/styles.d.ts +3 -3
- package/dist/Button/shared/types.d.ts +1 -1
- package/dist/ButtonBase/ButtonBase.d.ts +4 -9
- package/dist/ButtonBase/ButtonBase.js +4 -11
- package/dist/Card/elements.d.ts +103 -109
- package/dist/Card/styles.d.ts +8 -8
- package/dist/Coachmark/index.d.ts +1 -1
- package/dist/ConnectedForm/utils.d.ts +1 -1
- package/dist/DatePicker/Calendar/Calendar.d.ts +9 -0
- package/dist/DatePicker/Calendar/Calendar.js +28 -0
- package/dist/DatePicker/Calendar/CalendarBody.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarBody.js +174 -0
- package/dist/DatePicker/Calendar/CalendarFooter.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarFooter.js +54 -0
- package/dist/DatePicker/Calendar/CalendarHeader.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarHeader.js +86 -0
- package/dist/DatePicker/Calendar/index.d.ts +6 -0
- package/dist/DatePicker/Calendar/index.js +5 -0
- package/dist/DatePicker/Calendar/types.d.ts +64 -0
- package/dist/DatePicker/Calendar/types.js +1 -0
- package/dist/DatePicker/Calendar/utils/dateGrid.d.ts +30 -0
- package/dist/DatePicker/Calendar/utils/dateGrid.js +87 -0
- package/dist/DatePicker/Calendar/utils/format.d.ts +45 -0
- package/dist/DatePicker/Calendar/utils/format.js +123 -0
- package/dist/DatePicker/Calendar/utils/index.d.ts +3 -0
- package/dist/DatePicker/Calendar/utils/index.js +3 -0
- package/dist/DatePicker/Calendar/utils/keyHandler.d.ts +13 -0
- package/dist/DatePicker/Calendar/utils/keyHandler.js +125 -0
- package/dist/DatePicker/Calendar/utils/validation.d.ts +13 -0
- package/dist/DatePicker/Calendar/utils/validation.js +23 -0
- package/dist/DatePicker/DatePicker.d.ts +8 -0
- package/dist/DatePicker/DatePicker.js +128 -0
- package/dist/DatePicker/DatePickerCalendar.d.ts +13 -0
- package/dist/DatePicker/DatePickerCalendar.js +135 -0
- package/dist/DatePicker/DatePickerContext.d.ts +11 -0
- package/dist/DatePicker/DatePickerContext.js +18 -0
- package/dist/DatePicker/DatePickerInput.d.ts +16 -0
- package/dist/DatePicker/DatePickerInput.js +137 -0
- package/dist/DatePicker/index.d.ts +13 -0
- package/dist/DatePicker/index.js +10 -0
- package/dist/DatePicker/translations.d.ts +3 -0
- package/dist/DatePicker/translations.js +8 -0
- package/dist/DatePicker/types.d.ts +93 -0
- package/dist/DatePicker/types.js +1 -0
- package/dist/DatePicker/utils.d.ts +5 -0
- package/dist/DatePicker/utils.js +90 -0
- package/dist/Disclosure/elements.d.ts +12 -18
- package/dist/FeatureShimmer/index.js +1 -1
- package/dist/FocusTrap/index.d.ts +2 -2
- package/dist/Form/SelectDropdown/SelectDropdown.js +1 -1
- package/dist/Form/SelectDropdown/elements/controls.js +2 -2
- package/dist/Form/SelectDropdown/elements/multi-value.js +2 -2
- package/dist/Form/SelectDropdown/types/internal.d.ts +2 -2
- package/dist/Form/elements/Form.d.ts +15 -15
- package/dist/Form/elements/FormGroup.d.ts +1 -1
- package/dist/GridForm/GridFormButtons/index.d.ts +4 -4
- package/dist/List/ListProvider.d.ts +1 -1
- package/dist/List/elements.d.ts +43 -45
- package/dist/Menu/MenuItem.js +6 -10
- package/dist/Menu/elements.d.ts +2 -2
- package/dist/Modals/elements.d.ts +1 -1
- package/dist/Pagination/AnimatedPaginationButtons.d.ts +32 -34
- package/dist/Pagination/EllipsisButton.d.ts +4 -4
- package/dist/Pagination/PaginationButton.d.ts +6 -6
- package/dist/Pagination/utils.d.ts +30 -32
- package/dist/Pagination/utils.js +11 -14
- package/dist/Popover/Popover.js +4 -4
- package/dist/Popover/types.d.ts +2 -3
- package/dist/PopoverContainer/PopoverContainer.js +13 -9
- package/dist/PopoverContainer/hooks.d.ts +4 -16
- package/dist/PopoverContainer/hooks.js +24 -31
- package/dist/PopoverContainer/types.d.ts +7 -3
- package/dist/Tabs/TabButton.d.ts +2 -2
- package/dist/Tabs/TabNavLink.d.ts +2 -2
- package/dist/Tag/elements.d.ts +8 -14
- package/dist/Tip/InfoTip/InfoTipButton.d.ts +4 -4
- package/dist/Tip/PreviewTip/elements.d.ts +6 -12
- package/dist/Tip/__tests__/helpers.d.ts +1 -1
- package/dist/Tip/shared/FloatingTip.js +2 -2
- package/dist/Tip/shared/types.d.ts +2 -2
- package/dist/Typography/Text.d.ts +1 -1
- package/dist/index.d.ts +1 -0
- package/dist/index.js +1 -0
- package/dist/utils/react.js +1 -2
- package/package.json +11 -11
package/dist/Card/styles.d.ts
CHANGED
|
@@ -9,14 +9,14 @@ export declare const patternFadeInOut: {
|
|
|
9
9
|
opacity: number;
|
|
10
10
|
transition: {
|
|
11
11
|
duration: number;
|
|
12
|
-
ease:
|
|
12
|
+
ease: string;
|
|
13
13
|
};
|
|
14
14
|
};
|
|
15
15
|
animate: {
|
|
16
16
|
opacity: number;
|
|
17
17
|
transition: {
|
|
18
18
|
duration: number;
|
|
19
|
-
ease:
|
|
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:
|
|
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:
|
|
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:
|
|
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:
|
|
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:
|
|
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:
|
|
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) =>
|
|
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,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,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,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,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;
|