@codecademy/gamut 68.1.3-alpha.a2160b.0 → 68.1.3-alpha.b88d4a.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.
@@ -3,19 +3,25 @@ import { css, states } from '@codecademy/gamut-styles';
3
3
  import { useCallback, useEffect, useMemo, useRef } from 'react';
4
4
  import * as React from 'react';
5
5
  import { TextButton } from '../../Button';
6
- import { getMonthGrid, isDateDisabled, isDateInRange, isSameDay } from './utils/dateGrid';
6
+ import { getDatesWithRow, getMonthGrid, isDateDisabled, isDateInRange, isSameDay } from './utils/dateGrid';
7
7
  import { getWeekdayNames } from './utils/format';
8
- import { getDatesWithRow, keyHandler } from './utils/keyHandler';
8
+ import { 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: "e12sl4cx1",
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,{"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 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    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                  <td key={`empty-${rowIndex}-${colIndex}`} role=\"gridcell\" />\n                );\n              }\n              const selected =\n                isSameDay(date, selectedDate) || isSameDay(date, endDate);\n              const inRange =\n                !!selectedDate &&\n                !!endDate &&\n                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                <td\n                  aria-selected={selected}\n                  key={date.getTime()}\n                  role=\"gridcell\"\n                >\n                  <DateButton\n                    disabled={disabled}\n                    isInRange={inRange}\n                    isSelected={selected}\n                    isToday={today}\n                    ref={(el) => setButtonRef(date, el as HTMLElement | null)}\n                    tabIndex={isFocused ? 0 : -1}\n                    variant=\"secondary\"\n                    onClick={() => onDateSelect(date)}\n                    onFocus={() => onFocusedDateChange?.(date)}\n                    onKeyDown={(e: React.KeyboardEvent) =>\n                      handleKeyDown(e, date)\n                    }\n                  >\n                    {date.getDate()}\n                  </DateButton>\n                </td>\n              );\n            })}\n          </tr>\n        ))}\n      </tbody>\n    </table>\n  );\n};\n"]} */");
18
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DatePicker/Calendar/CalendarBody.tsx"],"names":[],"mappings":"AAiBoB","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  getDatesWithRow,\n  getMonthGrid,\n  isDateDisabled,\n  isDateInRange,\n  isSameDay,\n} from './utils/dateGrid';\nimport { getWeekdayNames } from './utils/format';\nimport { 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  displayDate,\n  selectedDate,\n  endDate = null,\n  disabledDates = [],\n  onDateSelect,\n  locale,\n  weekStartsOn = 0,\n  labelledById,\n  focusedDate,\n  onFocusedDateChange,\n  onDisplayDateChange,\n  onEscapeKeyPress,\n  hasAdjacentMonthRight,\n  hasAdjacentMonthLeft,\n}) => {\n  const year = displayDate.getFullYear();\n  const month = displayDate.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        onDisplayDateChange,\n        hasAdjacentMonthRight,\n        hasAdjacentMonthLeft\n      ),\n    [\n      onFocusedDateChange,\n      datesWithRow,\n      month,\n      year,\n      disabledDates,\n      onDateSelect,\n      onEscapeKeyPress,\n      onDisplayDateChange,\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":"AA0BiB","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  getDatesWithRow,\n  getMonthGrid,\n  isDateDisabled,\n  isDateInRange,\n  isSameDay,\n} from './utils/dateGrid';\nimport { getWeekdayNames } from './utils/format';\nimport { 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  displayDate,\n  selectedDate,\n  endDate = null,\n  disabledDates = [],\n  onDateSelect,\n  locale,\n  weekStartsOn = 0,\n  labelledById,\n  focusedDate,\n  onFocusedDateChange,\n  onDisplayDateChange,\n  onEscapeKeyPress,\n  hasAdjacentMonthRight,\n  hasAdjacentMonthLeft,\n}) => {\n  const year = displayDate.getFullYear();\n  const month = displayDate.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        onDisplayDateChange,\n        hasAdjacentMonthRight,\n        hasAdjacentMonthLeft\n      ),\n    [\n      onFocusedDateChange,\n      datesWithRow,\n      month,\n      year,\n      disabledDates,\n      onDateSelect,\n      onEscapeKeyPress,\n      onDisplayDateChange,\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"
@@ -43,6 +49,12 @@ const DateButton = /*#__PURE__*/_styled(TextButton, {
43
49
  bg: 'background'
44
50
  }
45
51
  },
52
+ isRangeStart: {
53
+ borderRadiusRight: 'none'
54
+ },
55
+ isRangeEnd: {
56
+ borderRadiusLeft: 'none'
57
+ },
46
58
  isInRange: {
47
59
  bg: 'text-disabled',
48
60
  color: 'background',
@@ -65,9 +77,9 @@ const DateButton = /*#__PURE__*/_styled(TextButton, {
65
77
  }), css({
66
78
  fontWeight: 'base',
67
79
  width: '32px'
68
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DatePicker/Calendar/CalendarBody.tsx"],"names":[],"mappings":"AAyBmB","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 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    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                  <td key={`empty-${rowIndex}-${colIndex}`} role=\"gridcell\" />\n                );\n              }\n              const selected =\n                isSameDay(date, selectedDate) || isSameDay(date, endDate);\n              const inRange =\n                !!selectedDate &&\n                !!endDate &&\n                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                <td\n                  aria-selected={selected}\n                  key={date.getTime()}\n                  role=\"gridcell\"\n                >\n                  <DateButton\n                    disabled={disabled}\n                    isInRange={inRange}\n                    isSelected={selected}\n                    isToday={today}\n                    ref={(el) => setButtonRef(date, el as HTMLElement | null)}\n                    tabIndex={isFocused ? 0 : -1}\n                    variant=\"secondary\"\n                    onClick={() => onDateSelect(date)}\n                    onFocus={() => onFocusedDateChange?.(date)}\n                    onKeyDown={(e: React.KeyboardEvent) =>\n                      handleKeyDown(e, date)\n                    }\n                  >\n                    {date.getDate()}\n                  </DateButton>\n                </td>\n              );\n            })}\n          </tr>\n        ))}\n      </tbody>\n    </table>\n  );\n};\n"]} */");
80
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DatePicker/Calendar/CalendarBody.tsx"],"names":[],"mappings":"AAgCmB","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  getDatesWithRow,\n  getMonthGrid,\n  isDateDisabled,\n  isDateInRange,\n  isSameDay,\n} from './utils/dateGrid';\nimport { getWeekdayNames } from './utils/format';\nimport { 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  displayDate,\n  selectedDate,\n  endDate = null,\n  disabledDates = [],\n  onDateSelect,\n  locale,\n  weekStartsOn = 0,\n  labelledById,\n  focusedDate,\n  onFocusedDateChange,\n  onDisplayDateChange,\n  onEscapeKeyPress,\n  hasAdjacentMonthRight,\n  hasAdjacentMonthLeft,\n}) => {\n  const year = displayDate.getFullYear();\n  const month = displayDate.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        onDisplayDateChange,\n        hasAdjacentMonthRight,\n        hasAdjacentMonthLeft\n      ),\n    [\n      onFocusedDateChange,\n      datesWithRow,\n      month,\n      year,\n      disabledDates,\n      onDateSelect,\n      onEscapeKeyPress,\n      onDisplayDateChange,\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"]} */");
69
81
  export const CalendarBody = ({
70
- visibleDate,
82
+ displayDate,
71
83
  selectedDate,
72
84
  endDate = null,
73
85
  disabledDates = [],
@@ -77,13 +89,13 @@ export const CalendarBody = ({
77
89
  labelledById,
78
90
  focusedDate,
79
91
  onFocusedDateChange,
80
- onVisibleDateChange,
92
+ onDisplayDateChange,
81
93
  onEscapeKeyPress,
82
94
  hasAdjacentMonthRight,
83
95
  hasAdjacentMonthLeft
84
96
  }) => {
85
- const year = visibleDate.getFullYear();
86
- const month = visibleDate.getMonth();
97
+ const year = displayDate.getFullYear();
98
+ const month = displayDate.getMonth();
87
99
  const weeks = getMonthGrid(year, month, weekStartsOn);
88
100
  const weekdayLabels = getWeekdayNames('short', locale, weekStartsOn);
89
101
  const weekdayFullNames = getWeekdayNames('long', locale, weekStartsOn);
@@ -99,7 +111,7 @@ export const CalendarBody = ({
99
111
  useEffect(() => {
100
112
  if (focusTarget !== null) focusButton(focusTarget);
101
113
  }, [focusTarget, focusButton]);
102
- 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]);
114
+ const handleKeyDown = useCallback((e, date) => keyHandler(e, date, onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onDisplayDateChange, hasAdjacentMonthRight, hasAdjacentMonthLeft), [onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onDisplayDateChange, hasAdjacentMonthLeft, hasAdjacentMonthRight]);
103
115
  const setButtonRef = useCallback((date, el) => {
104
116
  const k = new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime();
105
117
  if (el) buttonRefs.current.set(k, el);else buttonRefs.current.delete(k);
@@ -124,23 +136,26 @@ export const CalendarBody = ({
124
136
  /*#__PURE__*/
125
137
  // fix this error
126
138
  // eslint-disable-next-line react/no-array-index-key, jsx-a11y/control-has-associated-label
127
- _jsx("td", {
139
+ _jsx(DateCell, {
128
140
  role: "gridcell"
129
141
  }, `empty-${rowIndex}-${colIndex}`)
130
142
  );
131
143
  }
132
144
  const selected = isSameDay(date, selectedDate) || isSameDay(date, endDate);
133
- const inRange = !!selectedDate && !!endDate && isDateInRange(date, selectedDate, endDate);
145
+ const range = !!selectedDate && !!endDate;
146
+ const inRange = range && isDateInRange(date, selectedDate, endDate);
134
147
  const disabled = isDateDisabled(date, disabledDates);
135
148
  const today = isToday(date);
136
149
  // this is making the selected date a differnet color bc it is focused, look into further
137
150
  const isFocused = focusTarget !== null && isSameDay(date, focusTarget);
138
- return /*#__PURE__*/_jsx("td", {
151
+ return /*#__PURE__*/_jsx(DateCell, {
139
152
  "aria-selected": selected,
140
153
  role: "gridcell",
141
154
  children: /*#__PURE__*/_jsx(DateButton, {
142
155
  disabled: disabled,
143
156
  isInRange: inRange,
157
+ isRangeEnd: range && isSameDay(date, endDate),
158
+ isRangeStart: range && isSameDay(date, selectedDate),
144
159
  isSelected: selected,
145
160
  isToday: today,
146
161
  ref: el => setButtonRef(date, el),
@@ -1,6 +1,7 @@
1
1
  import * as React from 'react';
2
2
  import { FlexBox } from '../../Box';
3
3
  import { TextButton } from '../../Button';
4
+ import { DEFAULT_DATE_PICKER_TRANSLATIONS } from '../translations';
4
5
  import { getRelativeTodayLabel } from './utils/format';
5
6
 
6
7
  // function formatQuickActionLabel(action: QuickAction): string {
@@ -28,7 +29,7 @@ export const CalendarFooter = ({
28
29
  onClearDate,
29
30
  onTodayClick,
30
31
  locale,
31
- clearText,
32
+ clearText = DEFAULT_DATE_PICKER_TRANSLATIONS.clearText,
32
33
  disabled,
33
34
  showClearButton
34
35
  }) => {
@@ -6,9 +6,9 @@ import { Text } from '../../Typography';
6
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
- currentMonthYear,
10
- onCurrentMonthYearChange,
11
- secondMonthYear,
9
+ displayDate,
10
+ onDisplayDateChange,
11
+ secondDisplayDate,
12
12
  onLastMonthClick,
13
13
  onNextMonthClick,
14
14
  locale,
@@ -19,43 +19,62 @@ export const CalendarHeader = ({
19
19
  lastMonth
20
20
  } = getRelativeMonthLabels(locale);
21
21
  const handleLastMonth = () => {
22
- const lastMonth = new Date(currentMonthYear.getFullYear(), currentMonthYear.getMonth() - 1, 1);
23
- onCurrentMonthYearChange?.(lastMonth);
22
+ const lastMonth = new Date(displayDate.getFullYear(), displayDate.getMonth() - 1, 1);
23
+ onDisplayDateChange?.(lastMonth);
24
24
  onLastMonthClick?.();
25
25
  };
26
26
  const handleNextMonth = () => {
27
- const nextMonth = new Date(currentMonthYear.getFullYear(), currentMonthYear.getMonth() + 1, 1);
28
- onCurrentMonthYearChange?.(nextMonth);
27
+ const nextMonth = new Date(displayDate.getFullYear(), displayDate.getMonth() + 1, 1);
28
+ onDisplayDateChange?.(nextMonth);
29
29
  onNextMonthClick?.();
30
30
  };
31
31
  return /*#__PURE__*/_jsxs(FlexBox, {
32
32
  alignItems: "center",
33
- justifyContent: "space-between",
34
33
  pb: 16,
34
+ width: "100%",
35
35
  children: [/*#__PURE__*/_jsx(IconButton, {
36
36
  "aria-label": lastMonth,
37
37
  icon: MiniChevronLeftIcon,
38
38
  size: "small",
39
39
  tip: lastMonth,
40
40
  onClick: handleLastMonth
41
- }), /*#__PURE__*/_jsx(Text, {
42
- "aria-live": "polite",
43
- as: "h2",
44
- fontSize: 16,
45
- fontWeight: "title",
46
- id: headingId,
47
- children: formatMonthYear(currentMonthYear, locale)
48
- }), secondMonthYear && /*#__PURE__*/_jsx(Text, {
49
- "aria-live": "polite",
50
- as: "h2",
51
- display: {
52
- _: 'none',
53
- xs: 'initial'
41
+ }), /*#__PURE__*/_jsxs(FlexBox, {
42
+ flexGrow: 1,
43
+ gap: {
44
+ _: 0,
45
+ xs: secondDisplayDate ? 32 : 0
54
46
  },
55
- fontSize: 16,
56
- fontWeight: "title",
57
- id: headingId,
58
- children: formatMonthYear(secondMonthYear, locale)
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(displayDate, locale)
60
+ })
61
+ }), secondDisplayDate && /*#__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(secondDisplayDate, locale)
76
+ })
77
+ })]
59
78
  }), /*#__PURE__*/_jsx(IconButton, {
60
79
  "aria-label": nextMonth,
61
80
  icon: MiniChevronRightIcon,
@@ -3,11 +3,11 @@
3
3
  */
4
4
  export interface CalendarHeaderProps {
5
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;
6
+ displayDate: Date;
7
+ /** Called when the user navigates to a different month. Pass the new date (e.g. setDisplayDate) so the calendar updates. */
8
+ onDisplayDateChange: (newDate: Date) => void;
9
9
  /** Currently displayed second month and year (used for heading and prev/next range) */
10
- secondMonthYear?: Date;
10
+ secondDisplayDate?: 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. */
@@ -19,7 +19,9 @@ export interface CalendarHeaderProps {
19
19
  }
20
20
  export interface CalendarBodyProps {
21
21
  /** The month to display (typically first day of that month) */
22
- visibleDate: Date;
22
+ displayDate: Date;
23
+ /** Called when grid keyboard nav changes month (e.g. Page Up/Down). Pass setDisplayDate so the calendar updates. */
24
+ onDisplayDateChange: (newDate: Date) => void;
23
25
  /** Selected start date (single or range start) */
24
26
  selectedDate: Date | null;
25
27
  /** Selected end date (range only; undefined for single-date mode) */
@@ -38,8 +40,6 @@ export interface CalendarBodyProps {
38
40
  focusedDate: Date | null;
39
41
  /** Callback when focused date changes (e.g. arrow keys) */
40
42
  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
43
  /** Called when the escape key is pressed */
44
44
  onEscapeKeyPress?: () => void;
45
45
  /** When true (e.g. two-month view), arrow keys move focus to adjacent month without changing visible date. */
@@ -56,7 +56,7 @@ export interface CalendarFooterProps {
56
56
  disabled?: boolean;
57
57
  showClearButton?: boolean;
58
58
  locale?: string;
59
- clearText: string;
59
+ clearText?: string;
60
60
  onClearDate?: () => void;
61
61
  onTodayClick?: () => void;
62
62
  /** Max 3 quick actions (e.g. "7 days", "1 month") */
@@ -28,3 +28,8 @@ export declare const isDateInRange: (date: Date, start: Date | null, end: Date |
28
28
  * Check if `date` is in the `disabledDates` list (by calendar day).
29
29
  */
30
30
  export declare const isDateDisabled: (date: Date, disabledDates?: Date[]) => boolean;
31
+ /** Flat list of dates in grid order (row-major, non-null only) with row index for Home/End */
32
+ export declare const getDatesWithRow: (weeks: (Date | null)[][]) => {
33
+ date: Date;
34
+ rowIndex: number;
35
+ }[];
@@ -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);
@@ -90,4 +84,18 @@ export const isDateInRange = (date, start, end) => {
90
84
  */
91
85
  export const isDateDisabled = (date, disabledDates = []) => {
92
86
  return disabledDates.some(d => isSameDay(date, d));
87
+ };
88
+
89
+ /** Flat list of dates in grid order (row-major, non-null only) with row index for Home/End */
90
+ export const getDatesWithRow = weeks => {
91
+ const result = [];
92
+ weeks.forEach((week, rowIndex) => {
93
+ week.forEach(date => {
94
+ if (date !== null) result.push({
95
+ date,
96
+ rowIndex
97
+ });
98
+ });
99
+ });
100
+ return result;
93
101
  };
@@ -43,15 +43,3 @@ export declare const formatDateForInput: (date: Date, locale?: string) => string
43
43
  * Partial input like "1" or "2/15" returns null even though Date("1") would parse.
44
44
  */
45
45
  export declare const parseDateFromInput: (value: string, locale?: string) => Date | null;
46
- /**
47
- * Format a date range for the input (e.g. "2/15/2026 – 2/20/2026").
48
- */
49
- export declare const formatDateRangeForInput: (startDate: Date | null, endDate: Date | null, locale?: string) => string;
50
- /**
51
- * Parse a range string (e.g. "2/15/2026 – 2/20/2026") into { startDate, endDate }.
52
- * Returns null if invalid. Single date is allowed and yields startDate = endDate.
53
- */
54
- export declare const parseDateRangeFromInput: (value: string, locale?: string) => {
55
- startDate: Date;
56
- endDate: Date;
57
- } | null;
@@ -120,46 +120,4 @@ export const parseDateFromInput = (value, locale) => {
120
120
  const parts = trimmed.split(/[/-]/);
121
121
  if (parts.length >= 3) return parsed;
122
122
  return null;
123
- };
124
- const RANGE_SEPARATOR = ' – ';
125
-
126
- /**
127
- * Format a date range for the input (e.g. "2/15/2026 – 2/20/2026").
128
- */
129
- export const formatDateRangeForInput = (startDate, endDate, locale) => {
130
- if (!startDate && !endDate) return '';
131
- if (!startDate) return formatDateForInput(endDate, locale);
132
- if (!endDate) return formatDateForInput(startDate, locale);
133
- return `${formatDateForInput(startDate, locale)}${RANGE_SEPARATOR}${formatDateForInput(endDate, locale)}`;
134
- };
135
-
136
- /**
137
- * Parse a range string (e.g. "2/15/2026 – 2/20/2026") into { startDate, endDate }.
138
- * Returns null if invalid. Single date is allowed and yields startDate = endDate.
139
- */
140
- export const parseDateRangeFromInput = (value, locale) => {
141
- const trimmed = value.trim();
142
- if (!trimmed) return null;
143
- const parts = trimmed.split(RANGE_SEPARATOR).map(part => part.trim());
144
- if (parts.length === 1) {
145
- const date = parseDateFromInput(parts[0], locale);
146
- if (!date) return null;
147
- return {
148
- startDate: date,
149
- endDate: new Date(date)
150
- };
151
- }
152
- if (parts.length === 2) {
153
- const start = parseDateFromInput(parts[0], locale);
154
- const end = parseDateFromInput(parts[1], locale);
155
- if (!start || !end) return null;
156
- return start.getTime() <= end.getTime() ? {
157
- startDate: start,
158
- endDate: end
159
- } : {
160
- startDate: end,
161
- endDate: start
162
- };
163
- }
164
- return null;
165
123
  };
@@ -1,13 +1,4 @@
1
- /**
2
- * Clamp a day to the last day of the given month (e.g. Jan 31 -> Feb 28).
3
- */
4
- export declare const clampToMonth: (year: number, month: number, day: number) => Date;
5
- /** Flat list of dates in grid order (row-major, non-null only) with row index for Home/End */
6
- export declare const getDatesWithRow: (weeks: (Date | null)[][]) => {
7
- date: Date;
8
- rowIndex: number;
9
- }[];
10
1
  export declare const keyHandler: (e: React.KeyboardEvent, date: Date, onFocusedDateChange: (date: Date | null) => void, datesWithRow: {
11
2
  date: Date;
12
3
  rowIndex: number;
13
- }[], month: number, year: number, disabledDates: Date[], onDateSelect: (date: Date) => void, onEscapeKeyPress?: () => void, onVisibleDateChange?: ((newDate: Date) => void) | undefined, hasAdjacentMonthRight?: boolean, hasAdjacentMonthLeft?: boolean) => void;
4
+ }[], month: number, year: number, disabledDates: Date[], onDateSelect: (date: Date) => void, onEscapeKeyPress?: () => void, onDisplayDateChange?: ((newDate: Date) => void) | undefined, hasAdjacentMonthRight?: boolean, hasAdjacentMonthLeft?: boolean) => void;
@@ -3,25 +3,11 @@ import { isDateDisabled } from './dateGrid';
3
3
  /**
4
4
  * Clamp a day to the last day of the given month (e.g. Jan 31 -> Feb 28).
5
5
  */
6
- export const clampToMonth = (year, month, day) => {
6
+ const clampToMonth = (year, month, day) => {
7
7
  const last = new Date(year, month + 1, 0).getDate();
8
8
  return new Date(year, month, Math.min(day, last));
9
9
  };
10
-
11
- /** Flat list of dates in grid order (row-major, non-null only) with row index for Home/End */
12
- export const getDatesWithRow = weeks => {
13
- const result = [];
14
- weeks.forEach((week, rowIndex) => {
15
- week.forEach(date => {
16
- if (date !== null) result.push({
17
- date,
18
- rowIndex
19
- });
20
- });
21
- });
22
- return result;
23
- };
24
- export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onVisibleDateChange, hasAdjacentMonthRight, hasAdjacentMonthLeft) => {
10
+ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, year, disabledDates, onDateSelect, onEscapeKeyPress, onDisplayDateChange, hasAdjacentMonthRight, hasAdjacentMonthLeft) => {
25
11
  const idx = datesWithRow.findIndex(({
26
12
  date: dateWithRow
27
13
  }) => dateWithRow.getTime() === date.getTime());
@@ -31,7 +17,7 @@ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, ye
31
17
  const hasRight = !!hasAdjacentMonthRight;
32
18
  const hasLeft = !!hasAdjacentMonthLeft;
33
19
  let newDate = null;
34
- let newVisibleDate = null;
20
+ let newDisplayDate = null;
35
21
  switch (e.key) {
36
22
  case 'ArrowLeft':
37
23
  e.preventDefault();
@@ -41,7 +27,7 @@ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, ye
41
27
  const lastDayPrevMonth = new Date(year, month, 0);
42
28
  newDate = lastDayPrevMonth;
43
29
  if (!hasLeft) {
44
- newVisibleDate = new Date(year, month - 1, 1);
30
+ newDisplayDate = new Date(year, month - 1, 1);
45
31
  }
46
32
  }
47
33
  break;
@@ -52,7 +38,7 @@ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, ye
52
38
  } else {
53
39
  newDate = new Date(year, month + 1, 1);
54
40
  if (!hasRight) {
55
- newVisibleDate = new Date(year, month + 1, 1);
41
+ newDisplayDate = new Date(year, month + 1, 1);
56
42
  }
57
43
  }
58
44
  break;
@@ -62,7 +48,7 @@ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, ye
62
48
  newDate.setDate(newDate.getDate() - 7);
63
49
  if (newDate.getMonth() !== month || newDate.getFullYear() !== year) {
64
50
  if (!hasLeft) {
65
- newVisibleDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
51
+ newDisplayDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
66
52
  }
67
53
  }
68
54
  break;
@@ -72,7 +58,7 @@ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, ye
72
58
  newDate.setDate(newDate.getDate() + 7);
73
59
  if (newDate.getMonth() !== month || newDate.getFullYear() !== year) {
74
60
  if (!hasRight) {
75
- newVisibleDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
61
+ newDisplayDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
76
62
  }
77
63
  }
78
64
  break;
@@ -95,7 +81,7 @@ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, ye
95
81
  } else {
96
82
  newDate = clampToMonth(year, month + 1, day);
97
83
  }
98
- newVisibleDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
84
+ newDisplayDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
99
85
  break;
100
86
  case 'PageUp':
101
87
  e.preventDefault();
@@ -104,7 +90,7 @@ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, ye
104
90
  } else {
105
91
  newDate = clampToMonth(year, month - 1, day);
106
92
  }
107
- newVisibleDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
93
+ newDisplayDate = new Date(newDate.getFullYear(), newDate.getMonth(), 1);
108
94
  break;
109
95
  case 'Enter':
110
96
  case ' ':
@@ -120,6 +106,6 @@ export const keyHandler = (e, date, onFocusedDateChange, datesWithRow, month, ye
120
106
  }
121
107
  if (newDate !== null) {
122
108
  onFocusedDateChange(newDate);
123
- if (newVisibleDate !== null) onVisibleDateChange?.(newVisibleDate);
109
+ if (newDisplayDate !== null) onDisplayDateChange?.(newDisplayDate);
124
110
  }
125
111
  };
@@ -1,6 +1,6 @@
1
1
  import { MiniArrowRightIcon } from '@codecademy/gamut-icons';
2
2
  import { useCallback, useId, useMemo, useRef, useState } from 'react';
3
- import { FlexBox } from '../Box';
3
+ import { Box, FlexBox } from '../Box';
4
4
  import { PopoverContainer } from '../PopoverContainer';
5
5
  import { DatePickerCalendar } from './DatePickerCalendar';
6
6
  import { DatePickerProvider } from './DatePickerContext';
@@ -77,15 +77,16 @@ export const DatePicker = props => {
77
77
  children: [/*#__PURE__*/_jsx(FlexBox, {
78
78
  gap: 8,
79
79
  width: "fit-content",
80
- wrap: true,
81
80
  children: mode === 'range' ? /*#__PURE__*/_jsxs(_Fragment, {
82
81
  children: [/*#__PURE__*/_jsx(DatePickerInput, {
83
82
  label: props.startLabel,
84
83
  placeholder: placeholder,
85
84
  rangePart: "start",
86
85
  ref: inputRef
87
- }), /*#__PURE__*/_jsx(MiniArrowRightIcon, {
88
- alignSelf: "center"
86
+ }), /*#__PURE__*/_jsx(Box, {
87
+ alignSelf: "center",
88
+ mt: 32,
89
+ children: /*#__PURE__*/_jsx(MiniArrowRightIcon, {})
89
90
  }), /*#__PURE__*/_jsx(DatePickerInput, {
90
91
  label: props.endLabel,
91
92
  placeholder: placeholder,
@@ -35,7 +35,7 @@ export const DatePickerCalendar = ({
35
35
  const isRange = mode === 'range';
36
36
  const endDate = isRange ? context.endDate : undefined;
37
37
  const firstOfMonth = date => new Date(date.getFullYear(), date.getMonth(), 1);
38
- const [visibleDate, setVisibleDate] = useState(() => firstOfMonth(startOrSelectedDate ?? new Date()));
38
+ const [displayDate, setDisplayDate] = useState(() => firstOfMonth(startOrSelectedDate ?? new Date()));
39
39
  const [focusedDate, setFocusedDate] = useState(() => startOrSelectedDate ?? endDate ?? new Date());
40
40
  const wasOpenRef = useRef(false);
41
41
 
@@ -47,7 +47,7 @@ export const DatePickerCalendar = ({
47
47
  if (!justOpened) return;
48
48
  const anchor = startOrSelectedDate ?? endDate;
49
49
  if (anchor) {
50
- setVisibleDate(firstOfMonth(anchor));
50
+ setDisplayDate(firstOfMonth(anchor));
51
51
  setFocusedDate(startOrSelectedDate ?? endDate ?? new Date());
52
52
  }
53
53
  }, [isCalendarOpen, startOrSelectedDate, endDate]);
@@ -61,42 +61,42 @@ export const DatePickerCalendar = ({
61
61
  };
62
62
  const handleClearDate = () => {
63
63
  setSelection(null);
64
- setFocusedDate(visibleDate);
64
+ setFocusedDate(displayDate);
65
65
  };
66
66
  const handleTodayClick = () => {
67
67
  const today = new Date();
68
68
  setSelection(today);
69
- setVisibleDate(firstOfMonth(today));
69
+ setDisplayDate(firstOfMonth(today));
70
70
  setFocusedDate(today);
71
71
  };
72
72
  const focusTarget = focusedDate ?? startOrSelectedDate ?? endDate ?? new Date();
73
73
  const addMonths = (date, n) => new Date(date.getFullYear(), date.getMonth() + n, 1);
74
- const secondMonthDate = addMonths(visibleDate, 1);
74
+ const secondMonthDate = addMonths(displayDate, 1);
75
75
  const isTwoMonthsVisible = useMedia(`(min-width: ${breakpoints.xs})`);
76
76
  return /*#__PURE__*/_jsxs(Calendar, {
77
77
  children: [/*#__PURE__*/_jsxs(Box, {
78
78
  p: 24,
79
79
  children: [/*#__PURE__*/_jsx(CalendarHeader, {
80
- currentMonthYear: visibleDate,
80
+ displayDate: displayDate,
81
81
  headingId: headingId,
82
82
  locale: locale,
83
- secondMonthYear: secondMonthDate,
84
- onCurrentMonthYearChange: setVisibleDate
83
+ secondDisplayDate: secondMonthDate,
84
+ onDisplayDateChange: setDisplayDate
85
85
  }), /*#__PURE__*/_jsxs(FlexBox, {
86
86
  children: [/*#__PURE__*/_jsx(CalendarBody, {
87
87
  disabledDates: disabledDates,
88
+ displayDate: displayDate,
88
89
  endDate: endDate,
89
90
  focusedDate: focusTarget,
90
91
  hasAdjacentMonthRight: isTwoMonthsVisible,
91
92
  labelledById: headingId,
92
93
  locale: locale,
93
94
  selectedDate: startOrSelectedDate,
94
- visibleDate: visibleDate,
95
95
  weekStartsOn: weekStartsOn,
96
96
  onDateSelect: onDateSelect,
97
+ onDisplayDateChange: setDisplayDate,
97
98
  onEscapeKeyPress: closeCalendar,
98
- onFocusedDateChange: setFocusedDate,
99
- onVisibleDateChange: setVisibleDate
99
+ onFocusedDateChange: setFocusedDate
100
100
  }), /*#__PURE__*/_jsx(Box, {
101
101
  display: {
102
102
  _: 'none',
@@ -108,18 +108,18 @@ export const DatePickerCalendar = ({
108
108
  },
109
109
  children: /*#__PURE__*/_jsx(CalendarBody, {
110
110
  disabledDates: disabledDates,
111
+ displayDate: secondMonthDate,
111
112
  endDate: endDate,
112
113
  focusedDate: focusTarget,
113
114
  hasAdjacentMonthLeft: isTwoMonthsVisible,
114
115
  labelledById: headingId,
115
116
  locale: locale,
116
117
  selectedDate: startOrSelectedDate,
117
- visibleDate: secondMonthDate,
118
118
  weekStartsOn: weekStartsOn,
119
119
  onDateSelect: onDateSelect,
120
+ onDisplayDateChange: setDisplayDate,
120
121
  onEscapeKeyPress: closeCalendar,
121
- onFocusedDateChange: setFocusedDate,
122
- onVisibleDateChange: setVisibleDate
122
+ onFocusedDateChange: setFocusedDate
123
123
  })
124
124
  })]
125
125
  })]
@@ -8,4 +8,4 @@ export declare const DatePickerProvider: import("react").Provider<DatePickerCont
8
8
  * Must be used inside a DatePicker. For composed layouts, use this to get
9
9
  * openCalendar, closeCalendar, isCalendarOpen, inputRef, calendarDialogId, etc.
10
10
  */
11
- export declare function useDatePicker(): DatePickerContextValueType;
11
+ export declare const useDatePicker: () => DatePickerContextValueType;
@@ -9,10 +9,10 @@ export const DatePickerProvider = DatePickerContext.Provider;
9
9
  * Must be used inside a DatePicker. For composed layouts, use this to get
10
10
  * openCalendar, closeCalendar, isCalendarOpen, inputRef, calendarDialogId, etc.
11
11
  */
12
- export function useDatePicker() {
12
+ export const useDatePicker = () => {
13
13
  const value = useContext(DatePickerContext);
14
14
  if (value == null) {
15
15
  throw new Error('useDatePickerContext must be used within a DatePicker.');
16
16
  }
17
17
  return value;
18
- }
18
+ };
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@codecademy/gamut",
3
3
  "description": "Styleguide & Component library for Codecademy",
4
- "version": "68.1.3-alpha.a2160b.0",
4
+ "version": "68.1.3-alpha.b88d4a.0",
5
5
  "author": "Codecademy Engineering <dev@codecademy.com>",
6
6
  "dependencies": {
7
7
  "@codecademy/gamut-icons": "9.57.0",
@@ -59,5 +59,5 @@
59
59
  "dist/**/[A-Z]**/[A-Z]*.js",
60
60
  "dist/**/[A-Z]**/index.js"
61
61
  ],
62
- "gitHead": "18146c945a8c1d5fbf8a8019dbadc5b193de5c9c"
62
+ "gitHead": "6e0aad55dfb52bc57b740ccd0fd447c93251e5df"
63
63
  }