@codecademy/gamut 68.1.3-alpha.bcf87d.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/DatePicker/Calendar/Calendar.d.ts +5 -7
- package/dist/DatePicker/Calendar/Calendar.js +24 -11
- package/dist/DatePicker/Calendar/CalendarBody.js +33 -14
- package/dist/DatePicker/Calendar/CalendarFooter.js +13 -19
- package/dist/DatePicker/Calendar/CalendarHeader.js +52 -29
- package/dist/DatePicker/Calendar/types.d.ts +9 -4
- package/dist/DatePicker/Calendar/utils/dateGrid.js +8 -14
- package/dist/DatePicker/Calendar/utils/format.d.ts +23 -17
- package/dist/DatePicker/Calendar/utils/format.js +64 -71
- package/dist/DatePicker/Calendar/utils/keyHandler.d.ts +1 -1
- package/dist/DatePicker/Calendar/utils/keyHandler.js +17 -8
- package/dist/DatePicker/DatePicker.d.ts +1 -1
- package/dist/DatePicker/DatePicker.js +22 -21
- package/dist/DatePicker/DatePickerCalendar.js +12 -4
- package/dist/DatePicker/DatePickerContext.d.ts +1 -1
- package/dist/DatePicker/DatePickerContext.js +2 -2
- package/dist/DatePicker/DatePickerInput.js +38 -27
- package/dist/DatePicker/translations.d.ts +3 -0
- package/dist/DatePicker/translations.js +8 -0
- package/dist/DatePicker/types.d.ts +18 -1
- package/dist/DatePicker/utils.d.ts +3 -1
- package/dist/DatePicker/utils.js +29 -10
- package/package.json +2 -2
|
@@ -1,11 +1,9 @@
|
|
|
1
|
-
|
|
1
|
+
import * as React from 'react';
|
|
2
2
|
/**
|
|
3
3
|
* Outer wrapper for the calendar (header + body + footer).
|
|
4
4
|
* Used by DatePickerCalendar to group the calendar content.
|
|
5
|
+
* Renders a CheckerDense pattern shadow at offset left 8, top 8.
|
|
5
6
|
*/
|
|
6
|
-
export declare const Calendar:
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
} & {
|
|
10
|
-
theme?: import("@emotion/react").Theme | undefined;
|
|
11
|
-
}, import("react").DetailedHTMLProps<import("react").HTMLAttributes<HTMLDivElement>, HTMLDivElement>, {}>;
|
|
7
|
+
export declare const Calendar: React.FC<{
|
|
8
|
+
children: React.ReactNode;
|
|
9
|
+
}>;
|
|
@@ -1,15 +1,28 @@
|
|
|
1
|
-
import
|
|
2
|
-
import
|
|
1
|
+
import { CheckerDense } from '@codecademy/gamut-patterns';
|
|
2
|
+
import * as React from 'react';
|
|
3
|
+
import { Box } from '../../Box';
|
|
4
|
+
|
|
3
5
|
/**
|
|
4
6
|
* Outer wrapper for the calendar (header + body + footer).
|
|
5
7
|
* Used by DatePickerCalendar to group the calendar content.
|
|
8
|
+
* Renders a CheckerDense pattern shadow at offset left 8, top 8.
|
|
6
9
|
*/
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
})(
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
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
|
+
});
|
|
@@ -4,18 +4,24 @@ import { useCallback, useEffect, useMemo, useRef } from 'react';
|
|
|
4
4
|
import * as React from 'react';
|
|
5
5
|
import { TextButton } from '../../Button';
|
|
6
6
|
import { getMonthGrid, isDateDisabled, isDateInRange, isSameDay } from './utils/dateGrid';
|
|
7
|
-
import {
|
|
7
|
+
import { getWeekdayNames } from './utils/format';
|
|
8
8
|
import { getDatesWithRow, keyHandler } from './utils/keyHandler';
|
|
9
9
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
10
10
|
const TableHeader = /*#__PURE__*/_styled("th", {
|
|
11
|
-
target: "
|
|
11
|
+
target: "e12sl4cx2",
|
|
12
12
|
label: "TableHeader"
|
|
13
13
|
})(css({
|
|
14
14
|
fontSize: 14,
|
|
15
15
|
fontWeight: 'base',
|
|
16
16
|
color: 'text-disabled',
|
|
17
17
|
textAlign: 'center'
|
|
18
|
-
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
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"]} */");
|
|
19
25
|
const DateButton = /*#__PURE__*/_styled(TextButton, {
|
|
20
26
|
target: "e12sl4cx0",
|
|
21
27
|
label: "DateButton"
|
|
@@ -26,7 +32,6 @@ const DateButton = /*#__PURE__*/_styled(TextButton, {
|
|
|
26
32
|
content: '""',
|
|
27
33
|
position: 'absolute',
|
|
28
34
|
bottom: 4,
|
|
29
|
-
left: '50%',
|
|
30
35
|
width: 4,
|
|
31
36
|
height: 4,
|
|
32
37
|
borderRadius: 'full',
|
|
@@ -44,6 +49,12 @@ const DateButton = /*#__PURE__*/_styled(TextButton, {
|
|
|
44
49
|
bg: 'background'
|
|
45
50
|
}
|
|
46
51
|
},
|
|
52
|
+
isRangeStart: {
|
|
53
|
+
borderRadiusRight: 'none'
|
|
54
|
+
},
|
|
55
|
+
isRangeEnd: {
|
|
56
|
+
borderRadiusLeft: 'none'
|
|
57
|
+
},
|
|
47
58
|
isInRange: {
|
|
48
59
|
bg: 'text-disabled',
|
|
49
60
|
color: 'background',
|
|
@@ -58,12 +69,15 @@ const DateButton = /*#__PURE__*/_styled(TextButton, {
|
|
|
58
69
|
},
|
|
59
70
|
disabled: {
|
|
60
71
|
color: 'text-disabled',
|
|
61
|
-
textDecoration: 'line-through'
|
|
72
|
+
textDecoration: 'line-through',
|
|
73
|
+
'&:hover': {
|
|
74
|
+
textDecoration: 'line-through'
|
|
75
|
+
}
|
|
62
76
|
}
|
|
63
77
|
}), css({
|
|
64
78
|
fontWeight: 'base',
|
|
65
79
|
width: '32px'
|
|
66
|
-
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
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"]} */");
|
|
67
81
|
export const CalendarBody = ({
|
|
68
82
|
visibleDate,
|
|
69
83
|
selectedDate,
|
|
@@ -76,17 +90,19 @@ export const CalendarBody = ({
|
|
|
76
90
|
focusedDate,
|
|
77
91
|
onFocusedDateChange,
|
|
78
92
|
onVisibleDateChange,
|
|
79
|
-
onEscapeKeyPress
|
|
93
|
+
onEscapeKeyPress,
|
|
94
|
+
hasAdjacentMonthRight,
|
|
95
|
+
hasAdjacentMonthLeft
|
|
80
96
|
}) => {
|
|
81
97
|
const year = visibleDate.getFullYear();
|
|
82
98
|
const month = visibleDate.getMonth();
|
|
83
99
|
const weeks = getMonthGrid(year, month, weekStartsOn);
|
|
84
|
-
const weekdayLabels =
|
|
85
|
-
const weekdayFullNames =
|
|
100
|
+
const weekdayLabels = getWeekdayNames('short', locale, weekStartsOn);
|
|
101
|
+
const weekdayFullNames = getWeekdayNames('long', locale, weekStartsOn);
|
|
86
102
|
const buttonRefs = useRef(new Map());
|
|
87
103
|
const datesWithRow = useMemo(() => getDatesWithRow(weeks), [weeks]);
|
|
88
104
|
const focusTarget = focusedDate ?? selectedDate;
|
|
89
|
-
const isToday = useCallback(
|
|
105
|
+
const isToday = useCallback(date => date !== null && isSameDay(date, new Date()), []);
|
|
90
106
|
const focusButton = useCallback(date => {
|
|
91
107
|
if (date === null) return;
|
|
92
108
|
const key = new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime();
|
|
@@ -95,7 +111,7 @@ export const CalendarBody = ({
|
|
|
95
111
|
useEffect(() => {
|
|
96
112
|
if (focusTarget !== null) focusButton(focusTarget);
|
|
97
113
|
}, [focusTarget, focusButton]);
|
|
98
|
-
const handleKeyDown = useCallback((e, date) => keyHandler(e, date, onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onVisibleDateChange), [onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onVisibleDateChange]);
|
|
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]);
|
|
99
115
|
const setButtonRef = useCallback((date, el) => {
|
|
100
116
|
const k = new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime();
|
|
101
117
|
if (el) buttonRefs.current.set(k, el);else buttonRefs.current.delete(k);
|
|
@@ -120,23 +136,26 @@ export const CalendarBody = ({
|
|
|
120
136
|
/*#__PURE__*/
|
|
121
137
|
// fix this error
|
|
122
138
|
// eslint-disable-next-line react/no-array-index-key, jsx-a11y/control-has-associated-label
|
|
123
|
-
_jsx(
|
|
139
|
+
_jsx(DateCell, {
|
|
124
140
|
role: "gridcell"
|
|
125
141
|
}, `empty-${rowIndex}-${colIndex}`)
|
|
126
142
|
);
|
|
127
143
|
}
|
|
128
144
|
const selected = isSameDay(date, selectedDate) || isSameDay(date, endDate);
|
|
129
|
-
const
|
|
145
|
+
const range = !!selectedDate && !!endDate;
|
|
146
|
+
const inRange = range && isDateInRange(date, selectedDate, endDate);
|
|
130
147
|
const disabled = isDateDisabled(date, disabledDates);
|
|
131
148
|
const today = isToday(date);
|
|
132
149
|
// this is making the selected date a differnet color bc it is focused, look into further
|
|
133
150
|
const isFocused = focusTarget !== null && isSameDay(date, focusTarget);
|
|
134
|
-
return /*#__PURE__*/_jsx(
|
|
151
|
+
return /*#__PURE__*/_jsx(DateCell, {
|
|
135
152
|
"aria-selected": selected,
|
|
136
153
|
role: "gridcell",
|
|
137
154
|
children: /*#__PURE__*/_jsx(DateButton, {
|
|
138
155
|
disabled: disabled,
|
|
139
156
|
isInRange: inRange,
|
|
157
|
+
isRangeEnd: range && isSameDay(date, endDate),
|
|
158
|
+
isRangeStart: range && isSameDay(date, selectedDate),
|
|
140
159
|
isSelected: selected,
|
|
141
160
|
isToday: today,
|
|
142
161
|
ref: el => setButtonRef(date, el),
|
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import * as React from 'react';
|
|
2
2
|
import { FlexBox } from '../../Box';
|
|
3
3
|
import { TextButton } from '../../Button';
|
|
4
|
-
import {
|
|
4
|
+
import { getRelativeTodayLabel } from './utils/format';
|
|
5
|
+
|
|
5
6
|
// function formatQuickActionLabel(action: QuickAction): string {
|
|
6
7
|
// const { num, timePeriod } = action;
|
|
7
8
|
// const period =
|
|
@@ -22,23 +23,15 @@ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
|
22
23
|
// : 'years';
|
|
23
24
|
// return `${num} ${period}`;
|
|
24
25
|
// }
|
|
25
|
-
|
|
26
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
26
27
|
export const CalendarFooter = ({
|
|
27
28
|
onClearDate,
|
|
28
29
|
onTodayClick,
|
|
29
|
-
|
|
30
|
-
|
|
30
|
+
locale,
|
|
31
|
+
clearText,
|
|
32
|
+
disabled,
|
|
33
|
+
showClearButton
|
|
31
34
|
}) => {
|
|
32
|
-
const handleClearDate = () => {
|
|
33
|
-
onSelectedDateChange(null);
|
|
34
|
-
onClearDate?.();
|
|
35
|
-
};
|
|
36
|
-
const handleTodayClick = () => {
|
|
37
|
-
const today = new Date();
|
|
38
|
-
onSelectedDateChange(today);
|
|
39
|
-
onCurrentMonthYearChange(new Date(today.getFullYear(), today.getMonth(), 1));
|
|
40
|
-
onTodayClick?.();
|
|
41
|
-
};
|
|
42
35
|
// const actions = quickActions.slice(0, 3);
|
|
43
36
|
|
|
44
37
|
return /*#__PURE__*/_jsxs(FlexBox, {
|
|
@@ -46,14 +39,15 @@ export const CalendarFooter = ({
|
|
|
46
39
|
borderTop: 1,
|
|
47
40
|
justifyContent: "space-between",
|
|
48
41
|
p: 12,
|
|
49
|
-
children: [/*#__PURE__*/_jsx(TextButton, {
|
|
50
|
-
|
|
51
|
-
|
|
42
|
+
children: [showClearButton && /*#__PURE__*/_jsx(TextButton, {
|
|
43
|
+
disabled: disabled,
|
|
44
|
+
onClick: () => onClearDate?.(),
|
|
45
|
+
children: clearText
|
|
52
46
|
}), /*#__PURE__*/_jsx(FlexBox, {
|
|
53
47
|
gap: 32,
|
|
54
48
|
children: /*#__PURE__*/_jsx(TextButton, {
|
|
55
|
-
onClick:
|
|
56
|
-
children:
|
|
49
|
+
onClick: () => onTodayClick?.(),
|
|
50
|
+
children: getRelativeTodayLabel(locale)
|
|
57
51
|
})
|
|
58
52
|
})]
|
|
59
53
|
});
|
|
@@ -3,21 +3,25 @@ import * as React from 'react';
|
|
|
3
3
|
import { FlexBox } from '../../Box';
|
|
4
4
|
import { IconButton } from '../../Button';
|
|
5
5
|
import { Text } from '../../Typography';
|
|
6
|
-
import { formatMonthYear } from './utils/format';
|
|
6
|
+
import { formatMonthYear, getRelativeMonthLabels } from './utils/format';
|
|
7
7
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
8
8
|
export const CalendarHeader = ({
|
|
9
9
|
currentMonthYear,
|
|
10
10
|
onCurrentMonthYearChange,
|
|
11
11
|
secondMonthYear,
|
|
12
|
-
|
|
12
|
+
onLastMonthClick,
|
|
13
13
|
onNextMonthClick,
|
|
14
14
|
locale,
|
|
15
15
|
headingId
|
|
16
16
|
}) => {
|
|
17
|
-
const
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
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?.();
|
|
21
25
|
};
|
|
22
26
|
const handleNextMonth = () => {
|
|
23
27
|
const nextMonth = new Date(currentMonthYear.getFullYear(), currentMonthYear.getMonth() + 1, 1);
|
|
@@ -26,37 +30,56 @@ export const CalendarHeader = ({
|
|
|
26
30
|
};
|
|
27
31
|
return /*#__PURE__*/_jsxs(FlexBox, {
|
|
28
32
|
alignItems: "center",
|
|
29
|
-
justifyContent: "space-between",
|
|
30
33
|
pb: 16,
|
|
34
|
+
width: "100%",
|
|
31
35
|
children: [/*#__PURE__*/_jsx(IconButton, {
|
|
32
|
-
"aria-label":
|
|
36
|
+
"aria-label": lastMonth,
|
|
33
37
|
icon: MiniChevronLeftIcon,
|
|
34
38
|
size: "small",
|
|
35
|
-
tip:
|
|
36
|
-
onClick:
|
|
37
|
-
}), /*#__PURE__*/
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
id: headingId,
|
|
43
|
-
children: formatMonthYear(currentMonthYear, locale)
|
|
44
|
-
}), secondMonthYear && /*#__PURE__*/_jsx(Text, {
|
|
45
|
-
"aria-live": "polite",
|
|
46
|
-
as: "h2",
|
|
47
|
-
display: {
|
|
48
|
-
_: 'none',
|
|
49
|
-
xs: 'initial'
|
|
39
|
+
tip: lastMonth,
|
|
40
|
+
onClick: handleLastMonth
|
|
41
|
+
}), /*#__PURE__*/_jsxs(FlexBox, {
|
|
42
|
+
flexGrow: 1,
|
|
43
|
+
gap: {
|
|
44
|
+
_: 0,
|
|
45
|
+
xs: secondMonthYear ? 32 : 0
|
|
50
46
|
},
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
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
|
+
})]
|
|
55
78
|
}), /*#__PURE__*/_jsx(IconButton, {
|
|
56
|
-
"aria-label":
|
|
79
|
+
"aria-label": nextMonth,
|
|
57
80
|
icon: MiniChevronRightIcon,
|
|
58
81
|
size: "small",
|
|
59
|
-
tip:
|
|
82
|
+
tip: nextMonth,
|
|
60
83
|
onClick: handleNextMonth
|
|
61
84
|
})]
|
|
62
85
|
});
|
|
@@ -9,7 +9,7 @@ export interface CalendarHeaderProps {
|
|
|
9
9
|
/** Currently displayed second month and year (used for heading and prev/next range) */
|
|
10
10
|
secondMonthYear?: Date;
|
|
11
11
|
/** Called after navigating to previous month; use for click tracking. */
|
|
12
|
-
|
|
12
|
+
onLastMonthClick?: () => void;
|
|
13
13
|
/** Called after navigating to next month; use for click tracking. */
|
|
14
14
|
onNextMonthClick?: () => void;
|
|
15
15
|
/** Locale for month/year formatting (e.g. 'en-US') */
|
|
@@ -42,6 +42,10 @@ export interface CalendarBodyProps {
|
|
|
42
42
|
onVisibleDateChange: (newDate: Date) => void;
|
|
43
43
|
/** Called when the escape key is pressed */
|
|
44
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;
|
|
45
49
|
}
|
|
46
50
|
export interface QuickAction {
|
|
47
51
|
num: number;
|
|
@@ -49,11 +53,12 @@ export interface QuickAction {
|
|
|
49
53
|
onClick: () => void;
|
|
50
54
|
}
|
|
51
55
|
export interface CalendarFooterProps {
|
|
56
|
+
disabled?: boolean;
|
|
57
|
+
showClearButton?: boolean;
|
|
58
|
+
locale?: string;
|
|
59
|
+
clearText: string;
|
|
52
60
|
onClearDate?: () => void;
|
|
53
61
|
onTodayClick?: () => void;
|
|
54
|
-
/** Called when the user navigates to a different month. Pass the new date (e.g. setVisibleDate) so the calendar updates. */
|
|
55
|
-
onSelectedDateChange: (newDate: Date | null) => void;
|
|
56
|
-
onCurrentMonthYearChange: (newDate: Date) => void;
|
|
57
62
|
/** Max 3 quick actions (e.g. "7 days", "1 month") */
|
|
58
63
|
quickActions?: QuickAction[];
|
|
59
64
|
}
|
|
@@ -8,8 +8,8 @@ const DAYS_PER_WEEK = 7;
|
|
|
8
8
|
/**
|
|
9
9
|
* Normalize to start of day in local time for comparison.
|
|
10
10
|
*/
|
|
11
|
-
const
|
|
12
|
-
return new Date(date.getFullYear(), date.getMonth(), date.getDate());
|
|
11
|
+
const normalizeDate = date => {
|
|
12
|
+
return new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime();
|
|
13
13
|
};
|
|
14
14
|
|
|
15
15
|
/**
|
|
@@ -37,16 +37,10 @@ export const getMonthGrid = (year, month, weekStartsOn = 0) => {
|
|
|
37
37
|
const daysInMonth = last.getDate();
|
|
38
38
|
const weeks = [];
|
|
39
39
|
let currentWeek = [];
|
|
40
|
-
|
|
41
|
-
// Pad start of first week with nulls
|
|
42
|
-
// eslint-disable-next-line no-plusplus
|
|
43
|
-
for (let i = 0; i < firstDayOfWeek; i++) {
|
|
40
|
+
for (let i = 0; i < firstDayOfWeek; i += 1) {
|
|
44
41
|
currentWeek.push(null);
|
|
45
42
|
}
|
|
46
|
-
|
|
47
|
-
// fix these
|
|
48
|
-
// eslint-disable-next-line no-plusplus
|
|
49
|
-
for (let day = 1; day <= daysInMonth; day++) {
|
|
43
|
+
for (let day = 1; day <= daysInMonth; day += 1) {
|
|
50
44
|
currentWeek.push(new Date(year, month, day));
|
|
51
45
|
if (currentWeek.length === DAYS_PER_WEEK) {
|
|
52
46
|
weeks.push(currentWeek);
|
|
@@ -69,7 +63,7 @@ export const getMonthGrid = (year, month, weekStartsOn = 0) => {
|
|
|
69
63
|
*/
|
|
70
64
|
export const isSameDay = (a, b) => {
|
|
71
65
|
if (a === null || b === null) return false;
|
|
72
|
-
return
|
|
66
|
+
return normalizeDate(a) === normalizeDate(b);
|
|
73
67
|
};
|
|
74
68
|
|
|
75
69
|
/**
|
|
@@ -77,9 +71,9 @@ export const isSameDay = (a, b) => {
|
|
|
77
71
|
*/
|
|
78
72
|
export const isDateInRange = (date, start, end) => {
|
|
79
73
|
if (start === null) return false;
|
|
80
|
-
const normalizedDateTime =
|
|
81
|
-
const normalizedStartDateTime =
|
|
82
|
-
const normalizedEndDateTime = end !== null ?
|
|
74
|
+
const normalizedDateTime = normalizeDate(date);
|
|
75
|
+
const normalizedStartDateTime = normalizeDate(start);
|
|
76
|
+
const normalizedEndDateTime = end !== null ? normalizeDate(end) : normalizedStartDateTime;
|
|
83
77
|
const low = Math.min(normalizedStartDateTime, normalizedEndDateTime);
|
|
84
78
|
const high = Math.max(normalizedStartDateTime, normalizedEndDateTime);
|
|
85
79
|
return normalizedDateTime > low && normalizedDateTime < high;
|
|
@@ -1,20 +1,38 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Date formatting for the calendar using Intl.DateTimeFormat.
|
|
3
3
|
*/
|
|
4
|
+
/**
|
|
5
|
+
* Capitalize the first character of a string using the locale; rest unchanged (e.g. "next month" → "Next month").
|
|
6
|
+
*/
|
|
7
|
+
export declare const capitalizeFirst: (str: string, locale?: string) => string;
|
|
4
8
|
/**
|
|
5
9
|
* Format month and year for the calendar header (e.g. "February 2026").
|
|
6
10
|
*/
|
|
7
11
|
export declare const formatMonthYear: (date: Date, locale?: string) => string;
|
|
8
12
|
/**
|
|
9
|
-
* Get
|
|
13
|
+
* Get weekday names for column headers or abbr attributes.
|
|
10
14
|
* Order depends on weekStartsOn: 0 = Sunday first, 1 = Monday first.
|
|
15
|
+
* @param format - 'short' for abbreviated (e.g. "Su", "Mo"), 'long' for full (e.g. "Sunday", "Monday")
|
|
16
|
+
*/
|
|
17
|
+
export declare const getWeekdayNames: (format: 'short' | 'long', locale?: string, weekStartsOn?: 0 | 1) => string[];
|
|
18
|
+
/**
|
|
19
|
+
* Get localized "next month" and "previous month" labels for calendar nav.
|
|
20
|
+
* Uses Intl.RelativeTimeFormat with numeric: "auto" (e.g. "next month", "last month").
|
|
21
|
+
*/
|
|
22
|
+
export declare const getRelativeMonthLabels: (locale?: string) => {
|
|
23
|
+
nextMonth: string;
|
|
24
|
+
lastMonth: string;
|
|
25
|
+
};
|
|
26
|
+
/**
|
|
27
|
+
* Get localized "today" label (e.g. "today").
|
|
11
28
|
*/
|
|
12
|
-
export declare const
|
|
29
|
+
export declare const getRelativeTodayLabel: (locale?: string) => string;
|
|
13
30
|
/**
|
|
14
|
-
* Get
|
|
15
|
-
*
|
|
31
|
+
* Get the locale's short date format pattern (e.g. "MM/DD/YYYY" for en-US,
|
|
32
|
+
* "DD/MM/YYYY" for en-GB). Uses Intl.DateTimeFormat formatToParts to infer
|
|
33
|
+
* order and separators. Useful for parsing or building locale-aware inputs.
|
|
16
34
|
*/
|
|
17
|
-
export declare const
|
|
35
|
+
export declare const getDateFormatPattern: (locale?: string) => string;
|
|
18
36
|
/**
|
|
19
37
|
* Format a date for display in the date picker input (e.g. "2/15/2026").
|
|
20
38
|
*/
|
|
@@ -25,15 +43,3 @@ export declare const formatDateForInput: (date: Date, locale?: string) => string
|
|
|
25
43
|
* Partial input like "1" or "2/15" returns null even though Date("1") would parse.
|
|
26
44
|
*/
|
|
27
45
|
export declare const parseDateFromInput: (value: string, locale?: string) => Date | null;
|
|
28
|
-
/**
|
|
29
|
-
* Format a date range for the input (e.g. "2/15/2026 – 2/20/2026").
|
|
30
|
-
*/
|
|
31
|
-
export declare const formatDateRangeForInput: (startDate: Date | null, endDate: Date | null, locale?: string) => string;
|
|
32
|
-
/**
|
|
33
|
-
* Parse a range string (e.g. "2/15/2026 – 2/20/2026") into { startDate, endDate }.
|
|
34
|
-
* Returns null if invalid. Single date is allowed and yields startDate = endDate.
|
|
35
|
-
*/
|
|
36
|
-
export declare const parseDateRangeFromInput: (value: string, locale?: string) => {
|
|
37
|
-
startDate: Date;
|
|
38
|
-
endDate: Date;
|
|
39
|
-
} | null;
|