@codecademy/gamut 68.1.2 → 68.1.3-alpha.77d8dc.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (46) hide show
  1. package/dist/ConnectedForm/utils.d.ts +1 -1
  2. package/dist/DatePicker/Calendar/Calendar.d.ts +9 -0
  3. package/dist/DatePicker/Calendar/Calendar.js +28 -0
  4. package/dist/DatePicker/Calendar/CalendarBody.d.ts +3 -0
  5. package/dist/DatePicker/Calendar/CalendarBody.js +155 -0
  6. package/dist/DatePicker/Calendar/CalendarFooter.d.ts +3 -0
  7. package/dist/DatePicker/Calendar/CalendarFooter.js +54 -0
  8. package/dist/DatePicker/Calendar/CalendarHeader.d.ts +3 -0
  9. package/dist/DatePicker/Calendar/CalendarHeader.js +67 -0
  10. package/dist/DatePicker/Calendar/index.d.ts +6 -0
  11. package/dist/DatePicker/Calendar/index.js +5 -0
  12. package/dist/DatePicker/Calendar/types.d.ts +60 -0
  13. package/dist/DatePicker/Calendar/types.js +1 -0
  14. package/dist/DatePicker/Calendar/utils/dateGrid.d.ts +30 -0
  15. package/dist/DatePicker/Calendar/utils/dateGrid.js +93 -0
  16. package/dist/DatePicker/Calendar/utils/format.d.ts +61 -0
  17. package/dist/DatePicker/Calendar/utils/format.js +184 -0
  18. package/dist/DatePicker/Calendar/utils/index.d.ts +3 -0
  19. package/dist/DatePicker/Calendar/utils/index.js +3 -0
  20. package/dist/DatePicker/Calendar/utils/keyHandler.d.ts +13 -0
  21. package/dist/DatePicker/Calendar/utils/keyHandler.js +116 -0
  22. package/dist/DatePicker/Calendar/utils/validation.d.ts +13 -0
  23. package/dist/DatePicker/Calendar/utils/validation.js +23 -0
  24. package/dist/DatePicker/DatePicker.d.ts +8 -0
  25. package/dist/DatePicker/DatePicker.js +147 -0
  26. package/dist/DatePicker/DatePickerCalendar.d.ts +13 -0
  27. package/dist/DatePicker/DatePickerCalendar.js +130 -0
  28. package/dist/DatePicker/DatePickerContext.d.ts +11 -0
  29. package/dist/DatePicker/DatePickerContext.js +18 -0
  30. package/dist/DatePicker/DatePickerInput.d.ts +16 -0
  31. package/dist/DatePicker/DatePickerInput.js +135 -0
  32. package/dist/DatePicker/index.d.ts +13 -0
  33. package/dist/DatePicker/index.js +10 -0
  34. package/dist/DatePicker/translations.d.ts +3 -0
  35. package/dist/DatePicker/translations.js +4 -0
  36. package/dist/DatePicker/types.d.ts +85 -0
  37. package/dist/DatePicker/types.js +1 -0
  38. package/dist/DatePicker/utils.d.ts +3 -0
  39. package/dist/DatePicker/utils.js +71 -0
  40. package/dist/FocusTrap/index.d.ts +2 -2
  41. package/dist/List/elements.d.ts +1 -1
  42. package/dist/PopoverContainer/PopoverContainer.js +3 -1
  43. package/dist/PopoverContainer/types.d.ts +5 -0
  44. package/dist/index.d.ts +1 -0
  45. package/dist/index.js +1 -0
  46. package/package.json +2 -2
@@ -0,0 +1,18 @@
1
+ import { createContext, useContext } from 'react';
2
+ export const DatePickerContext = /*#__PURE__*/createContext(null);
3
+
4
+ /** Provider component; DatePicker uses this to set the context value. */
5
+ export const DatePickerProvider = DatePickerContext.Provider;
6
+
7
+ /**
8
+ * Returns the DatePicker context value (shared state and callbacks).
9
+ * Must be used inside a DatePicker. For composed layouts, use this to get
10
+ * openCalendar, closeCalendar, isCalendarOpen, inputRef, calendarDialogId, etc.
11
+ */
12
+ export function useDatePicker() {
13
+ const value = useContext(DatePickerContext);
14
+ if (value == null) {
15
+ throw new Error('useDatePickerContext must be used within a DatePicker.');
16
+ }
17
+ return value;
18
+ }
@@ -0,0 +1,16 @@
1
+ import { ComponentProps } from 'react';
2
+ import { Input } from '../Form/inputs/Input';
3
+ /**
4
+ * Props for DatePickerInput. When used inside DatePicker, only overrides (e.g. placeholder, label).
5
+ * In range mode, use rangePart to bind to start or end date. When outside DatePicker, pass value, onChange, etc.
6
+ */
7
+ export type DatePickerInputProps = Omit<ComponentProps<typeof Input>, 'type' | 'icon'> & {
8
+ /** In range mode: which part of the range this input edits. Omit for single-date or combined display. */
9
+ rangePart?: 'start' | 'end';
10
+ };
11
+ /**
12
+ * Date input. When inside DatePicker: owns local input value state and syncs to
13
+ * shared selectedDate via context on blur/parse; opens calendar on click/arrow down.
14
+ * When outside DatePicker: fully controlled by props.
15
+ */
16
+ export declare const DatePickerInput: import("react").ForwardRefExoticComponent<Omit<DatePickerInputProps, "ref"> & import("react").RefAttributes<HTMLInputElement>>;
@@ -0,0 +1,135 @@
1
+ import { MiniCalendarIcon } from '@codecademy/gamut-icons';
2
+ import { forwardRef, useEffect, useId, useRef, useState } from 'react';
3
+ import { FormGroup } from '../Form/elements/FormGroup';
4
+ import { Input } from '../Form/inputs/Input';
5
+ import { formatDateForInput, getDateFormatPattern, parseDateFromInput } from './Calendar/utils/format';
6
+ import { useDatePicker } from './DatePickerContext';
7
+
8
+ /**
9
+ * Props for DatePickerInput. When used inside DatePicker, only overrides (e.g. placeholder, label).
10
+ * In range mode, use rangePart to bind to start or end date. When outside DatePicker, pass value, onChange, etc.
11
+ */
12
+ import { jsx as _jsx } from "react/jsx-runtime";
13
+ /**
14
+ * Date input. When inside DatePicker: owns local input value state and syncs to
15
+ * shared selectedDate via context on blur/parse; opens calendar on click/arrow down.
16
+ * When outside DatePicker: fully controlled by props.
17
+ */
18
+ export const DatePickerInput = /*#__PURE__*/forwardRef(({
19
+ placeholder,
20
+ label,
21
+ rangePart,
22
+ ...rest
23
+ }, ref) => {
24
+ const context = useDatePicker();
25
+ // do we want to do this or just throw an error?
26
+ // if (context == null) {
27
+ // return (
28
+ // <Input
29
+ // {...rest}
30
+ // icon={CalendarIcon}
31
+ // placeholder={placeholder ?? 'MM/DD/YYYY'}
32
+ // ref={ref}
33
+ // type="text"
34
+ // />
35
+ // );
36
+ // }
37
+ if (context == null) {
38
+ throw new Error('DatePickerInput must be used inside a DatePicker (it reads shared state from context).');
39
+ }
40
+ const {
41
+ mode,
42
+ startOrSelectedDate,
43
+ setSelection,
44
+ openCalendar,
45
+ locale,
46
+ isCalendarOpen,
47
+ calendarDialogId
48
+ } = context;
49
+ const isRange = mode === 'range';
50
+ const inputID = useId();
51
+ const inputId = `datepicker-input-${inputID.replace(/:/g, '')}`;
52
+
53
+ // Range with two inputs: each input binds to one part. Single or range combined: one value.
54
+ const boundDate = isRange && rangePart === 'end' ? context.endDate : startOrSelectedDate;
55
+ const formattedValue = boundDate != null ? formatDateForInput(boundDate, locale) : '';
56
+ const [inputValue, setInputValue] = useState(() => formattedValue);
57
+ const isInputFocusedRef = useRef(false);
58
+
59
+ // Sync input from shared state. Skip when focused so we don't overwrite while typing.
60
+ useEffect(() => {
61
+ if (!isInputFocusedRef.current) {
62
+ setInputValue(formattedValue);
63
+ }
64
+ }, [formattedValue]);
65
+
66
+ /** Apply raw input string to selection state. Returns formatted string if parsed so caller can sync input (e.g. on blur). */
67
+ const applyValueToSelection = raw => {
68
+ const trimmed = raw.trim();
69
+ if (!trimmed) {
70
+ if (isRange && rangePart) {
71
+ if (rangePart === 'start') setSelection(null, context.endDate);else setSelection(startOrSelectedDate, null);
72
+ } else setSelection(null);
73
+ return undefined;
74
+ }
75
+ const parsed = parseDateFromInput(trimmed, locale);
76
+ if (!parsed) return undefined;
77
+ if (isRange && rangePart) {
78
+ if (rangePart === 'start') setSelection(parsed, context.endDate);else setSelection(startOrSelectedDate, parsed);
79
+ } else setSelection(parsed);
80
+ return formatDateForInput(parsed, locale);
81
+ };
82
+ const handleChange = e => {
83
+ const raw = e.target.value;
84
+ setInputValue(raw);
85
+ applyValueToSelection(raw);
86
+ };
87
+ const handleBlur = () => {
88
+ isInputFocusedRef.current = false;
89
+ const formatted = applyValueToSelection(inputValue.trim());
90
+ if (formatted) setInputValue(formatted);else if (inputValue.trim()) setInputValue(formattedValue);
91
+ };
92
+ const handleKeyDown = e => {
93
+ if (e.key === 'ArrowDown' || e.key === 'Down') {
94
+ e.preventDefault();
95
+ openCalendar();
96
+ }
97
+ };
98
+ const handleOpenCalendar = () => {
99
+ openCalendar();
100
+ };
101
+ const defaultLabel = isRange && rangePart === 'end' ? 'End date' : isRange ? 'Start date' : 'Date';
102
+ return /*#__PURE__*/_jsx(FormGroup, {
103
+ htmlFor: inputId,
104
+ isSoloField: true // should probaly be based on a prop
105
+ ,
106
+ label: label ?? defaultLabel,
107
+ pb: 0,
108
+ spacing: "tight",
109
+ width: "170px",
110
+ children: /*#__PURE__*/_jsx(Input, {
111
+ ...rest,
112
+ "aria-autocomplete": "none",
113
+ "aria-controls": calendarDialogId,
114
+ "aria-expanded": isCalendarOpen,
115
+ "aria-haspopup": "dialog",
116
+ icon: () => /*#__PURE__*/_jsx(MiniCalendarIcon, {
117
+ size: 16
118
+ }),
119
+ id: inputId,
120
+ placeholder: placeholder ?? getDateFormatPattern(locale),
121
+ ref: ref,
122
+ role: "combobox",
123
+ type: "text",
124
+ value: inputValue,
125
+ onBlur: handleBlur,
126
+ onChange: handleChange,
127
+ onClick: handleOpenCalendar,
128
+ onFocus: () => {
129
+ isInputFocusedRef.current = true;
130
+ if (isRange && rangePart) context.setActiveRangePart(rangePart);
131
+ },
132
+ onKeyDown: handleKeyDown
133
+ })
134
+ });
135
+ });
@@ -0,0 +1,13 @@
1
+ /**
2
+ * DatePicker – Single-date and range picker with input + calendar popover.
3
+ */
4
+ export type { DatePickerContextValue, DatePickerProps, DatePickerRangeProps, DatePickerSingleProps, } from './types';
5
+ export { DatePicker } from './DatePicker';
6
+ export { DatePickerContext, DatePickerProvider, useDatePicker, } from './DatePickerContext';
7
+ export { DatePickerCalendar } from './DatePickerCalendar';
8
+ export type { DatePickerCalendarProps } from './DatePickerCalendar';
9
+ export { DatePickerInput } from './DatePickerInput';
10
+ export type { DatePickerInputProps } from './DatePickerInput';
11
+ export { Calendar, CalendarHeader, CalendarBody, CalendarFooter, } from './Calendar';
12
+ export type { CalendarHeaderProps, CalendarBodyProps, CalendarFooterProps, QuickAction, } from './Calendar/types';
13
+ export * from './Calendar/utils';
@@ -0,0 +1,10 @@
1
+ /**
2
+ * DatePicker – Single-date and range picker with input + calendar popover.
3
+ */
4
+
5
+ export { DatePicker } from './DatePicker';
6
+ export { DatePickerContext, DatePickerProvider, useDatePicker } from './DatePickerContext';
7
+ export { DatePickerCalendar } from './DatePickerCalendar';
8
+ export { DatePickerInput } from './DatePickerInput';
9
+ export { Calendar, CalendarHeader, CalendarBody, CalendarFooter } from './Calendar';
10
+ export * from './Calendar/utils';
@@ -0,0 +1,3 @@
1
+ import { DatePickerTranslations } from './types';
2
+ /** Default UI strings; pass translations prop to override. */
3
+ export declare const DEFAULT_DATE_PICKER_TRANSLATIONS: Required<DatePickerTranslations>;
@@ -0,0 +1,4 @@
1
+ /** Default UI strings; pass translations prop to override. */
2
+ export const DEFAULT_DATE_PICKER_TRANSLATIONS = {
3
+ clear: 'Clear'
4
+ };
@@ -0,0 +1,85 @@
1
+ /**
2
+ * Public and internal types for the DatePicker (single-date and range).
3
+ */
4
+ /// <reference types="react" />
5
+ /** Result of custom validation; null means valid. */
6
+ export interface DatePickerValidationResult {
7
+ errorMessage: string;
8
+ errorType: string;
9
+ }
10
+ /** Shared props for all DatePicker modes. */
11
+ export interface DatePickerBaseProps {
12
+ /** Locale for formatting (e.g. 'en-US'). */
13
+ locale?: string;
14
+ /** Dates that are disabled (unselectable) in the calendar. */
15
+ disabledDates?: Date[];
16
+ /** When provided, only the provider is rendered and children compose Input + Calendar. */
17
+ children?: React.ReactNode;
18
+ /** Placeholder for the input. */
19
+ placeholder?: string;
20
+ /** Override UI strings (e.g. clear button). Merged with defaults. */
21
+ translations?: DatePickerTranslations;
22
+ }
23
+ /** Props for the DatePicker (single-date mode). */
24
+ export interface DatePickerSingleProps extends DatePickerBaseProps {
25
+ mode?: 'single';
26
+ /** Controlled selected date. */
27
+ selectedDate: Date | null;
28
+ /** Called when the user selects a date. */
29
+ setSelectedDate: (date: Date | null) => void;
30
+ /** Label for the input. */
31
+ label?: string;
32
+ }
33
+ /** Props for the DatePicker (range mode). */
34
+ export interface DatePickerRangeProps extends DatePickerBaseProps {
35
+ mode: 'range';
36
+ /** Controlled start date. */
37
+ startDate: Date | null;
38
+ /** Controlled end date. */
39
+ endDate: Date | null;
40
+ /** Called when the user changes the start date. */
41
+ setStartDate: (date: Date | null) => void;
42
+ /** Called when the user changes the end date. */
43
+ setEndDate: (date: Date | null) => void;
44
+ /** Label for the start date input. */
45
+ startLabel?: string;
46
+ /** Label for the end date input. */
47
+ endLabel?: string;
48
+ }
49
+ /** Props for the DatePicker provider / standalone component. */
50
+ export type DatePickerProps = DatePickerSingleProps | DatePickerRangeProps;
51
+ /** Which range input is active (focused); null = calendar drives both (selection mode). */
52
+ export type ActiveRangePart = 'start' | 'end' | null;
53
+ /** Optional translations for DatePicker UI strings. Pass to override defaults. */
54
+ export interface DatePickerTranslations {
55
+ /** Label for the clear date button (default: "Clear"). */
56
+ clear?: string;
57
+ }
58
+ /** Shared state provided by DatePicker via context. */
59
+ export interface DatePickerBaseContextValue {
60
+ isCalendarOpen: boolean;
61
+ openCalendar: () => void;
62
+ closeCalendar: () => void;
63
+ locale?: string;
64
+ disabledDates: Date[];
65
+ calendarDialogId: string;
66
+ /** UI string overrides (e.g. clear button). */
67
+ translations: Required<DatePickerTranslations>;
68
+ /** Start date (range) or selected date (single). */
69
+ startOrSelectedDate: Date | null;
70
+ /** Set selection. Single: (date). Range: (start, end). */
71
+ setSelection: (startOrSelectedDate: Date | null, endDate?: Date | null) => void;
72
+ }
73
+ export interface DatePickerSingleContextValue extends DatePickerBaseContextValue {
74
+ mode: 'single';
75
+ }
76
+ export interface DatePickerRangeContextValue extends DatePickerBaseContextValue {
77
+ mode: 'range';
78
+ /** Range only: end date. */
79
+ endDate: Date | null;
80
+ /** Range only: which input is active (start/end focused); null = selection mode. */
81
+ activeRangePart: ActiveRangePart;
82
+ /** Range only: set which input is active (e.g. when input receives focus). */
83
+ setActiveRangePart: (part: ActiveRangePart) => void;
84
+ }
85
+ export type DatePickerContextValue = DatePickerSingleContextValue | DatePickerRangeContextValue;
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,3 @@
1
+ import { ActiveRangePart } from './types';
2
+ export declare const handleDateSelectSingle: (date: Date, selectedDate: Date | null, setSelection: (date: Date | null) => void) => void;
3
+ export declare const handleDateSelectRange: (date: Date, activeRangePart: ActiveRangePart, startDate: Date | null, endDate: Date | null, setSelection: (startDate: Date | null, endDate?: Date | null) => void) => void;
@@ -0,0 +1,71 @@
1
+ export const handleDateSelectSingle = (date, selectedDate, setSelection) => {
2
+ // If clicked date is the same as Start Date: Clear Start Date
3
+ if (selectedDate && date.getTime() === selectedDate.getTime()) {
4
+ setSelection(null);
5
+ return;
6
+ }
7
+ // If clicked date is not the same as Start Date: Set Start Date to clicked date
8
+ setSelection(date);
9
+ };
10
+ export const handleDateSelectRange = (date, activeRangePart, startDate, endDate, setSelection) => {
11
+ // Range mode: field targeting (start or end input was focused)
12
+ if (activeRangePart === 'start') {
13
+ if (date.getTime() === startDate?.getTime()) {
14
+ setSelection(null, endDate);
15
+ return;
16
+ }
17
+ const newEnd = endDate != null && date.getTime() <= endDate.getTime() ? endDate : null;
18
+ setSelection(date, newEnd);
19
+ return;
20
+ }
21
+ if (activeRangePart === 'end') {
22
+ if (date.getTime() === endDate?.getTime()) {
23
+ setSelection(startDate, null);
24
+ return;
25
+ }
26
+ const newStart = startDate != null && date.getTime() >= startDate.getTime() ? startDate : null;
27
+ setSelection(newStart, date);
28
+ return;
29
+ }
30
+
31
+ // Range selection mode (no field focused: calendar drives both)
32
+ if (startDate && endDate) {
33
+ // if start date is end date and is clicked, clears everything
34
+ if (startDate.getTime() === endDate.getTime() && date.getTime() === startDate.getTime()) {
35
+ setSelection(null, null);
36
+ return;
37
+ }
38
+ // if clicked on start date, end date becomes start date
39
+ if (date.getTime() === startDate.getTime()) {
40
+ setSelection(endDate, null);
41
+ return;
42
+ }
43
+ // if clicked on end date, clears end date and start remains
44
+ if (date.getTime() === endDate.getTime()) {
45
+ setSelection(startDate, null);
46
+ return;
47
+ }
48
+ // If clicked date > Start: Updates End Date to new date (Start remains)
49
+ if (date.getTime() > startDate.getTime()) {
50
+ setSelection(startDate, date);
51
+ return;
52
+ }
53
+ // If clicked date < Start: Updates Start Date to new date (End remains) - extends range to the left
54
+
55
+ setSelection(date, endDate);
56
+ return;
57
+ }
58
+ // Start is Set, End is Empty
59
+ if (startDate && !endDate) {
60
+ // If clicked date < Start: Restarts selection with clicked date as new Start
61
+ if (date.getTime() < startDate.getTime()) {
62
+ setSelection(date, null);
63
+ }
64
+ // If clicked date > Start: Sets it as End Date
65
+ else {
66
+ setSelection(startDate, date);
67
+ }
68
+ return;
69
+ }
70
+ setSelection(date, null);
71
+ };
@@ -23,8 +23,8 @@ export interface FocusTrapProps extends WithChildrenProp {
23
23
  */
24
24
  allowPageInteraction?: boolean;
25
25
  /**
26
- * Passthrough for react-focus-on library props
26
+ * Passthrough for react-focus-on library props (partial; only override what you need).
27
27
  */
28
- focusOnProps?: ReactFocusOnProps;
28
+ focusOnProps?: Partial<ReactFocusOnProps>;
29
29
  }
30
30
  export declare const FocusTrap: React.FC<FocusTrapProps>;
@@ -388,7 +388,7 @@ export declare const HeaderRowEl: import("@emotion/styled").StyledComponent<{
388
388
  theme?: import("@emotion/react").Theme | undefined;
389
389
  as?: import("react").ElementType<any, keyof import("react").JSX.IntrinsicElements> | undefined;
390
390
  } & HeaderProps, Pick<import("react").DetailedHTMLProps<import("react").HTMLAttributes<HTMLTableRowElement>, HTMLTableRowElement>, "slot" | "style" | "title" | "dir" | "children" | "className" | "aria-hidden" | "onAnimationStart" | "onDragStart" | "onDragEnd" | "onDrag" | keyof import("react").ClassAttributes<HTMLDivElement> | "defaultChecked" | "defaultValue" | "suppressContentEditableWarning" | "suppressHydrationWarning" | "accessKey" | "autoCapitalize" | "autoFocus" | "contentEditable" | "contextMenu" | "draggable" | "enterKeyHint" | "hidden" | "id" | "lang" | "nonce" | "spellCheck" | "tabIndex" | "translate" | "radioGroup" | "role" | "about" | "content" | "datatype" | "inlist" | "prefix" | "property" | "rel" | "resource" | "rev" | "typeof" | "vocab" | "autoCorrect" | "autoSave" | "itemProp" | "itemScope" | "itemType" | "itemID" | "itemRef" | "results" | "security" | "unselectable" | "inputMode" | "is" | "exportparts" | "part" | "aria-activedescendant" | "aria-atomic" | "aria-autocomplete" | "aria-braillelabel" | "aria-brailleroledescription" | "aria-busy" | "aria-checked" | "aria-colcount" | "aria-colindex" | "aria-colindextext" | "aria-colspan" | "aria-controls" | "aria-current" | "aria-describedby" | "aria-description" | "aria-details" | "aria-disabled" | "aria-dropeffect" | "aria-errormessage" | "aria-expanded" | "aria-flowto" | "aria-grabbed" | "aria-haspopup" | "aria-invalid" | "aria-keyshortcuts" | "aria-label" | "aria-labelledby" | "aria-level" | "aria-live" | "aria-modal" | "aria-multiline" | "aria-multiselectable" | "aria-orientation" | "aria-owns" | "aria-placeholder" | "aria-posinset" | "aria-pressed" | "aria-readonly" | "aria-relevant" | "aria-required" | "aria-roledescription" | "aria-rowcount" | "aria-rowindex" | "aria-rowindextext" | "aria-rowspan" | "aria-selected" | "aria-setsize" | "aria-sort" | "aria-valuemax" | "aria-valuemin" | "aria-valuenow" | "aria-valuetext" | "dangerouslySetInnerHTML" | "onCopy" | "onCopyCapture" | "onCut" | "onCutCapture" | "onPaste" | "onPasteCapture" | "onCompositionEnd" | "onCompositionEndCapture" | "onCompositionStart" | "onCompositionStartCapture" | "onCompositionUpdate" | "onCompositionUpdateCapture" | "onFocus" | "onFocusCapture" | "onBlur" | "onBlurCapture" | "onChange" | "onChangeCapture" | "onBeforeInput" | "onBeforeInputCapture" | "onInput" | "onInputCapture" | "onReset" | "onResetCapture" | "onSubmit" | "onSubmitCapture" | "onInvalid" | "onInvalidCapture" | "onLoad" | "onLoadCapture" | "onError" | "onErrorCapture" | "onKeyDown" | "onKeyDownCapture" | "onKeyPress" | "onKeyPressCapture" | "onKeyUp" | "onKeyUpCapture" | "onAbort" | "onAbortCapture" | "onCanPlay" | "onCanPlayCapture" | "onCanPlayThrough" | "onCanPlayThroughCapture" | "onDurationChange" | "onDurationChangeCapture" | "onEmptied" | "onEmptiedCapture" | "onEncrypted" | "onEncryptedCapture" | "onEnded" | "onEndedCapture" | "onLoadedData" | "onLoadedDataCapture" | "onLoadedMetadata" | "onLoadedMetadataCapture" | "onLoadStart" | "onLoadStartCapture" | "onPause" | "onPauseCapture" | "onPlay" | "onPlayCapture" | "onPlaying" | "onPlayingCapture" | "onProgress" | "onProgressCapture" | "onRateChange" | "onRateChangeCapture" | "onSeeked" | "onSeekedCapture" | "onSeeking" | "onSeekingCapture" | "onStalled" | "onStalledCapture" | "onSuspend" | "onSuspendCapture" | "onTimeUpdate" | "onTimeUpdateCapture" | "onVolumeChange" | "onVolumeChangeCapture" | "onWaiting" | "onWaitingCapture" | "onAuxClick" | "onAuxClickCapture" | "onClick" | "onClickCapture" | "onContextMenu" | "onContextMenuCapture" | "onDoubleClick" | "onDoubleClickCapture" | "onDragCapture" | "onDragEndCapture" | "onDragEnter" | "onDragEnterCapture" | "onDragExit" | "onDragExitCapture" | "onDragLeave" | "onDragLeaveCapture" | "onDragOver" | "onDragOverCapture" | "onDragStartCapture" | "onDrop" | "onDropCapture" | "onMouseDown" | "onMouseDownCapture" | "onMouseEnter" | "onMouseLeave" | "onMouseMove" | "onMouseMoveCapture" | "onMouseOut" | "onMouseOutCapture" | "onMouseOver" | "onMouseOverCapture" | "onMouseUp" | "onMouseUpCapture" | "onSelect" | "onSelectCapture" | "onTouchCancel" | "onTouchCancelCapture" | "onTouchEnd" | "onTouchEndCapture" | "onTouchMove" | "onTouchMoveCapture" | "onTouchStart" | "onTouchStartCapture" | "onPointerDown" | "onPointerDownCapture" | "onPointerMove" | "onPointerMoveCapture" | "onPointerUp" | "onPointerUpCapture" | "onPointerCancel" | "onPointerCancelCapture" | "onPointerEnter" | "onPointerLeave" | "onPointerOver" | "onPointerOverCapture" | "onPointerOut" | "onPointerOutCapture" | "onGotPointerCapture" | "onGotPointerCaptureCapture" | "onLostPointerCapture" | "onLostPointerCaptureCapture" | "onScroll" | "onScrollCapture" | "onWheel" | "onWheelCapture" | "onAnimationStartCapture" | "onAnimationEnd" | "onAnimationEndCapture" | "onAnimationIteration" | "onAnimationIterationCapture" | "onTransitionEnd" | "onTransitionEndCapture">, {}>;
391
- declare const columnType: (props: import("@codecademy/variance/dist/types/config").VariantProps<"type", false | "header" | "select" | "content" | "orderedHeader" | "control" | "tableControl" | "expand" | "expandControl"> & {
391
+ declare const columnType: (props: import("@codecademy/variance/dist/types/config").VariantProps<"type", false | "header" | "select" | "content" | "control" | "orderedHeader" | "tableControl" | "expand" | "expandControl"> & {
392
392
  theme?: import("@emotion/react").Theme | undefined;
393
393
  }) => import("@codecademy/variance").CSSObject;
394
394
  declare const columnJustify: (props: import("@codecademy/variance/dist/types/config").VariantProps<"justify", false | "left" | "right"> & {
@@ -16,7 +16,7 @@ const PopoverContent = /*#__PURE__*/_styled("div", {
16
16
  transform: {
17
17
  property: 'transform'
18
18
  }
19
- })), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/PopoverContainer/PopoverContainer.tsx"],"names":[],"mappings":"AAiBuB","file":"../../src/PopoverContainer/PopoverContainer.tsx","sourcesContent":["import { system } from '@codecademy/gamut-styles';\nimport { variance } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react';\nimport * as React from 'react';\nimport { useWindowScroll, useWindowSize } from 'react-use';\n\nimport { BodyPortal } from '../BodyPortal';\nimport { FocusTrap } from '../FocusTrap';\nimport {\n  useResizingParentEffect,\n  useScrollingParents,\n  useScrollingParentsEffect,\n} from './hooks';\nimport { ContainerState, PopoverContainerProps } from './types';\nimport { getContainers, getPosition, isOutOfView } from './utils';\n\nconst PopoverContent = styled.div(\n  variance.compose(\n    system.positioning,\n    variance.create({\n      transform: {\n        property: 'transform',\n      },\n    })\n  )\n);\n\nexport const PopoverContainer: React.FC<PopoverContainerProps> = ({\n  alignment = 'bottom-left',\n  offset = 20,\n  y = 0,\n  x = 0,\n  invertAxis,\n  inline = false,\n  isOpen,\n  onRequestClose,\n  targetRef,\n  allowPageInteraction,\n  closeOnViewportExit = false,\n  ...rest\n}) => {\n  const popoverRef = useRef<HTMLDivElement>(null);\n  const hasRequestedCloseRef = useRef(false);\n  const onRequestCloseRef = useRef(onRequestClose);\n  const { width: winW, height: winH } = useWindowSize();\n  const { x: winX, y: winY } = useWindowScroll();\n  const [containers, setContainers] = useState<ContainerState>();\n  const [targetRect, setTargetRect] = useState<DOMRect>();\n  const parent = containers?.parent;\n\n  // Memoize scrolling parents to avoid expensive DOM traversals\n  const scrollingParents = useScrollingParents(\n    targetRef as React.RefObject<HTMLElement | null>\n  );\n\n  // Keep onRequestClose ref up to date\n  useEffect(() => {\n    onRequestCloseRef.current = onRequestClose;\n  }, [onRequestClose]);\n\n  // Detect RTL direction from the target element and watch for attribute changes so the\n  // position recalculates when changes occur\n  const [isRtl, setIsRtl] = useState(false);\n  useEffect(() => {\n    const checkDirection = () => {\n      const target = targetRef?.current;\n      const el = target instanceof Element ? target : document.documentElement;\n      setIsRtl(getComputedStyle(el).direction === 'rtl');\n    };\n\n    checkDirection();\n\n    const observer = new MutationObserver(checkDirection);\n    observer.observe(document.documentElement, {\n      attributes: true,\n      attributeFilter: ['dir'],\n      subtree: true,\n    });\n    return () => observer.disconnect();\n  }, [targetRef]);\n\n  const popoverPosition = useMemo(() => {\n    if (parent !== undefined) {\n      return getPosition({\n        alignment,\n        container: parent,\n        invertAxis,\n        isRtl,\n        offset,\n        x,\n        y,\n      });\n    }\n    return { styles: {}, physicalStyles: undefined };\n  }, [parent, x, y, offset, alignment, invertAxis, isRtl]);\n\n  useEffect(() => {\n    const target = targetRef?.current;\n    if (!target) return;\n    setContainers(getContainers(target, inline, { x: winX, y: winY }));\n  }, [targetRef, inline, winW, winH, winX, winY, targetRect]);\n\n  // Update target rectangle when window size/scroll changes\n  useEffect(() => {\n    setTargetRect(targetRef?.current?.getBoundingClientRect());\n  }, [targetRef, isOpen, winW, winH, winX, winY]);\n\n  // Update target rectangle when parent size/scroll changes\n  const updateTargetPosition = useCallback(\n    (rect?: DOMRect) => {\n      const target = targetRef?.current;\n      if (!target) return;\n\n      const newRect = rect || target.getBoundingClientRect();\n      setTargetRect(newRect);\n\n      const currentScrollX =\n        window.pageXOffset || document.documentElement.scrollLeft;\n      const currentScrollY =\n        window.pageYOffset || document.documentElement.scrollTop;\n\n      setContainers(\n        getContainers(target, inline, { x: currentScrollX, y: currentScrollY })\n      );\n    },\n    [targetRef, inline]\n  );\n\n  useScrollingParentsEffect(targetRef, updateTargetPosition);\n\n  useResizingParentEffect(targetRef, setTargetRect);\n\n  // Handle closeOnViewportExit with cached scrolling parents for performance\n  useEffect(() => {\n    if (!closeOnViewportExit) return;\n\n    const rect = targetRect || containers?.viewport;\n    if (!rect) return;\n\n    const isOut = isOutOfView(\n      rect,\n      targetRef?.current as HTMLElement,\n      scrollingParents\n    );\n\n    if (isOut && !hasRequestedCloseRef.current) {\n      hasRequestedCloseRef.current = true;\n      onRequestCloseRef.current?.();\n    } else if (!isOut) {\n      hasRequestedCloseRef.current = false;\n    }\n  }, [\n    targetRect,\n    containers?.viewport,\n    targetRef,\n    closeOnViewportExit,\n    scrollingParents,\n  ]);\n  /**\n   * Allows targetRef to be or contain a button that toggles the popover open and closed.\n   * Without this check it would toggle closed then back open immediately.\n   *\n   */\n  const handleClickOutside = useCallback(\n    (e: MouseEvent | TouchEvent) => {\n      const target = e.target as Node;\n      const targetElement = targetRef.current;\n\n      if (!targetElement) return;\n      if (targetElement.contains(target)) return;\n      if (popoverRef.current?.contains(target)) return;\n\n      // If we get here, it's a genuine outside click\n      onRequestClose?.();\n    },\n    [onRequestClose, targetRef]\n  );\n\n  /**\n   * Backup click outside handler for cases where FocusTrap detection might be interfered with\n   * by our own floating elements\n   */\n  const handleGlobalClickOutside = useCallback(\n    (e: MouseEvent) => {\n      const target = e.target as Node;\n      const targetElement = targetRef.current;\n\n      if (!targetElement || !isOpen) return;\n\n      if (\n        targetElement.contains(target) ||\n        popoverRef.current?.contains(target)\n      )\n        return;\n\n      // Check if the clicked element is within an Overlay component\n      const clickedElement = target as Element;\n      if (clickedElement.closest('[data-floating=\"overlay\"]')) {\n        return;\n      }\n\n      // Check if the clicked element is within another Popover or PopoverContainer\n      const isFloatingElement = clickedElement.closest(\n        '[data-floating=\"popover\"]'\n      );\n      if (\n        isFloatingElement &&\n        !popoverRef.current?.contains(isFloatingElement)\n      ) {\n        onRequestClose?.();\n        return;\n      }\n\n      onRequestClose?.();\n    },\n    [onRequestClose, targetRef, isOpen]\n  );\n\n  // Backup global click listener for when a Popover or PopoverContainer is open\n  useEffect(() => {\n    if (isOpen) {\n      // Use a small delay to ensure this doesn't interfere with the FocusTrap's own detection\n      const timeoutId = setTimeout(() => {\n        document.addEventListener('mousedown', handleGlobalClickOutside, true);\n      }, 50);\n\n      return () => {\n        clearTimeout(timeoutId);\n        document.removeEventListener(\n          'mousedown',\n          handleGlobalClickOutside,\n          true\n        );\n      };\n    }\n  }, [isOpen, handleGlobalClickOutside]);\n\n  if (!isOpen || !targetRef) return null;\n\n  const content = (\n    <FocusTrap\n      allowPageInteraction={inline || allowPageInteraction}\n      onClickOutside={handleClickOutside}\n      onEscapeKey={onRequestClose}\n    >\n      <PopoverContent\n        data-floating=\"popover\"\n        data-testid=\"popover-content-container\"\n        position=\"absolute\"\n        ref={popoverRef}\n        tabIndex={-1}\n        zIndex={inline ? 5 : 'initial'}\n        {...popoverPosition.styles}\n        /* Physical inline style for centered alignments (top/bottom) where\n           inset-inline-start would incorrectly flip the center point in RTL */\n        /* eslint-disable-next-line gamut/no-inline-style */\n        style={popoverPosition.physicalStyles}\n        {...rest}\n      />\n    </FocusTrap>\n  );\n\n  if (inline) return content;\n\n  return <BodyPortal>{content}</BodyPortal>;\n};\n"]} */");
19
+ })), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/PopoverContainer/PopoverContainer.tsx"],"names":[],"mappings":"AAiBuB","file":"../../src/PopoverContainer/PopoverContainer.tsx","sourcesContent":["import { system } from '@codecademy/gamut-styles';\nimport { variance } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react';\nimport * as React from 'react';\nimport { useWindowScroll, useWindowSize } from 'react-use';\n\nimport { BodyPortal } from '../BodyPortal';\nimport { FocusTrap } from '../FocusTrap';\nimport {\n  useResizingParentEffect,\n  useScrollingParents,\n  useScrollingParentsEffect,\n} from './hooks';\nimport { ContainerState, PopoverContainerProps } from './types';\nimport { getContainers, getPosition, isOutOfView } from './utils';\n\nconst PopoverContent = styled.div(\n  variance.compose(\n    system.positioning,\n    variance.create({\n      transform: {\n        property: 'transform',\n      },\n    })\n  )\n);\n\nexport const PopoverContainer: React.FC<PopoverContainerProps> = ({\n  alignment = 'bottom-left',\n  offset = 20,\n  y = 0,\n  x = 0,\n  invertAxis,\n  inline = false,\n  isOpen,\n  onRequestClose,\n  targetRef,\n  allowPageInteraction,\n  closeOnViewportExit = false,\n  focusOnProps,\n  ...rest\n}) => {\n  const popoverRef = useRef<HTMLDivElement>(null);\n  const hasRequestedCloseRef = useRef(false);\n  const onRequestCloseRef = useRef(onRequestClose);\n  const { width: winW, height: winH } = useWindowSize();\n  const { x: winX, y: winY } = useWindowScroll();\n  const [containers, setContainers] = useState<ContainerState>();\n  const [targetRect, setTargetRect] = useState<DOMRect>();\n  const parent = containers?.parent;\n\n  // Memoize scrolling parents to avoid expensive DOM traversals\n  const scrollingParents = useScrollingParents(\n    targetRef as React.RefObject<HTMLElement | null>\n  );\n\n  // Keep onRequestClose ref up to date\n  useEffect(() => {\n    onRequestCloseRef.current = onRequestClose;\n  }, [onRequestClose]);\n\n  // Detect RTL direction from the target element and watch for attribute changes so the\n  // position recalculates when changes occur\n  const [isRtl, setIsRtl] = useState(false);\n  useEffect(() => {\n    const checkDirection = () => {\n      const target = targetRef?.current;\n      const el = target instanceof Element ? target : document.documentElement;\n      setIsRtl(getComputedStyle(el).direction === 'rtl');\n    };\n\n    checkDirection();\n\n    const observer = new MutationObserver(checkDirection);\n    observer.observe(document.documentElement, {\n      attributes: true,\n      attributeFilter: ['dir'],\n      subtree: true,\n    });\n    return () => observer.disconnect();\n  }, [targetRef]);\n\n  const popoverPosition = useMemo(() => {\n    if (parent !== undefined) {\n      return getPosition({\n        alignment,\n        container: parent,\n        invertAxis,\n        isRtl,\n        offset,\n        x,\n        y,\n      });\n    }\n    return { styles: {}, physicalStyles: undefined };\n  }, [parent, x, y, offset, alignment, invertAxis, isRtl]);\n\n  useEffect(() => {\n    const target = targetRef?.current;\n    if (!target) return;\n    setContainers(getContainers(target, inline, { x: winX, y: winY }));\n  }, [targetRef, inline, winW, winH, winX, winY, targetRect]);\n\n  // Update target rectangle when window size/scroll changes\n  useEffect(() => {\n    setTargetRect(targetRef?.current?.getBoundingClientRect());\n  }, [targetRef, isOpen, winW, winH, winX, winY]);\n\n  // Update target rectangle when parent size/scroll changes\n  const updateTargetPosition = useCallback(\n    (rect?: DOMRect) => {\n      const target = targetRef?.current;\n      if (!target) return;\n\n      const newRect = rect || target.getBoundingClientRect();\n      setTargetRect(newRect);\n\n      const currentScrollX =\n        window.pageXOffset || document.documentElement.scrollLeft;\n      const currentScrollY =\n        window.pageYOffset || document.documentElement.scrollTop;\n\n      setContainers(\n        getContainers(target, inline, { x: currentScrollX, y: currentScrollY })\n      );\n    },\n    [targetRef, inline]\n  );\n\n  useScrollingParentsEffect(targetRef, updateTargetPosition);\n\n  useResizingParentEffect(targetRef, setTargetRect);\n\n  // Handle closeOnViewportExit with cached scrolling parents for performance\n  useEffect(() => {\n    if (!closeOnViewportExit) return;\n\n    const rect = targetRect || containers?.viewport;\n    if (!rect) return;\n\n    const isOut = isOutOfView(\n      rect,\n      targetRef?.current as HTMLElement,\n      scrollingParents\n    );\n\n    if (isOut && !hasRequestedCloseRef.current) {\n      hasRequestedCloseRef.current = true;\n      onRequestCloseRef.current?.();\n    } else if (!isOut) {\n      hasRequestedCloseRef.current = false;\n    }\n  }, [\n    targetRect,\n    containers?.viewport,\n    targetRef,\n    closeOnViewportExit,\n    scrollingParents,\n  ]);\n  /**\n   * Allows targetRef to be or contain a button that toggles the popover open and closed.\n   * Without this check it would toggle closed then back open immediately.\n   *\n   */\n  const handleClickOutside = useCallback(\n    (e: MouseEvent | TouchEvent) => {\n      const target = e.target as Node;\n      const targetElement = targetRef.current;\n\n      if (!targetElement) return;\n      if (targetElement.contains(target)) return;\n      if (popoverRef.current?.contains(target)) return;\n\n      // If we get here, it's a genuine outside click\n      onRequestClose?.();\n    },\n    [onRequestClose, targetRef]\n  );\n\n  /**\n   * Backup click outside handler for cases where FocusTrap detection might be interfered with\n   * by our own floating elements\n   */\n  const handleGlobalClickOutside = useCallback(\n    (e: MouseEvent) => {\n      const target = e.target as Node;\n      const targetElement = targetRef.current;\n\n      if (!targetElement || !isOpen) return;\n\n      if (\n        targetElement.contains(target) ||\n        popoverRef.current?.contains(target)\n      )\n        return;\n\n      // Check if the clicked element is within an Overlay component\n      const clickedElement = target as Element;\n      if (clickedElement.closest('[data-floating=\"overlay\"]')) {\n        return;\n      }\n\n      // Check if the clicked element is within another Popover or PopoverContainer\n      const isFloatingElement = clickedElement.closest(\n        '[data-floating=\"popover\"]'\n      );\n      if (\n        isFloatingElement &&\n        !popoverRef.current?.contains(isFloatingElement)\n      ) {\n        onRequestClose?.();\n        return;\n      }\n\n      onRequestClose?.();\n    },\n    [onRequestClose, targetRef, isOpen]\n  );\n\n  // Backup global click listener for when a Popover or PopoverContainer is open\n  useEffect(() => {\n    if (isOpen) {\n      // Use a small delay to ensure this doesn't interfere with the FocusTrap's own detection\n      const timeoutId = setTimeout(() => {\n        document.addEventListener('mousedown', handleGlobalClickOutside, true);\n      }, 50);\n\n      return () => {\n        clearTimeout(timeoutId);\n        document.removeEventListener(\n          'mousedown',\n          handleGlobalClickOutside,\n          true\n        );\n      };\n    }\n  }, [isOpen, handleGlobalClickOutside]);\n\n  if (!isOpen || !targetRef) return null;\n\n  const content = (\n    <FocusTrap\n      allowPageInteraction={inline || allowPageInteraction}\n      focusOnProps={focusOnProps}\n      onClickOutside={handleClickOutside}\n      onEscapeKey={onRequestClose}\n    >\n      <PopoverContent\n        data-floating=\"popover\"\n        data-testid=\"popover-content-container\"\n        position=\"absolute\"\n        ref={popoverRef}\n        tabIndex={-1}\n        zIndex={inline ? 5 : 'initial'}\n        {...popoverPosition.styles}\n        /* Physical inline style for centered alignments (top/bottom) where\n           inset-inline-start would incorrectly flip the center point in RTL */\n        /* eslint-disable-next-line gamut/no-inline-style */\n        style={popoverPosition.physicalStyles}\n        {...rest}\n      />\n    </FocusTrap>\n  );\n\n  if (inline) return content;\n\n  return <BodyPortal>{content}</BodyPortal>;\n};\n"]} */");
20
20
  export const PopoverContainer = ({
21
21
  alignment = 'bottom-left',
22
22
  offset = 20,
@@ -29,6 +29,7 @@ export const PopoverContainer = ({
29
29
  targetRef,
30
30
  allowPageInteraction,
31
31
  closeOnViewportExit = false,
32
+ focusOnProps,
32
33
  ...rest
33
34
  }) => {
34
35
  const popoverRef = useRef(null);
@@ -189,6 +190,7 @@ export const PopoverContainer = ({
189
190
  if (!isOpen || !targetRef) return null;
190
191
  const content = /*#__PURE__*/_jsx(FocusTrap, {
191
192
  allowPageInteraction: inline || allowPageInteraction,
193
+ focusOnProps: focusOnProps,
192
194
  onClickOutside: handleClickOutside,
193
195
  onEscapeKey: onRequestClose,
194
196
  children: /*#__PURE__*/_jsx(PopoverContent, {
@@ -65,4 +65,9 @@ export interface PopoverContainerProps extends PopoverAlignment, WithChildrenPro
65
65
  * Defaults to false.
66
66
  */
67
67
  closeOnViewportExit?: boolean;
68
+ /**
69
+ * Optional props passed to the internal FocusTrap (react-focus-on).
70
+ * Use e.g. { autoFocus: false, focusLock: false } to keep focus on the trigger when the popover opens.
71
+ */
72
+ focusOnProps?: Partial<import('../FocusTrap').FocusTrapProps['focusOnProps']>;
68
73
  }
package/dist/index.d.ts CHANGED
@@ -15,6 +15,7 @@ export * from './Card';
15
15
  export * from './Coachmark';
16
16
  export * from './ConnectedForm';
17
17
  export * from './ContentContainer';
18
+ export * from './DatePicker';
18
19
  export * from './DelayedRenderWrapper';
19
20
  export * from './Disclosure';
20
21
  export * from './DataList';
package/dist/index.js CHANGED
@@ -14,6 +14,7 @@ export * from './Card';
14
14
  export * from './Coachmark';
15
15
  export * from './ConnectedForm';
16
16
  export * from './ContentContainer';
17
+ export * from './DatePicker';
17
18
  export * from './DelayedRenderWrapper';
18
19
  export * from './Disclosure';
19
20
  export * from './DataList';
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.2",
4
+ "version": "68.1.3-alpha.77d8dc.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": "2f5ee4fd98708f6922e81293f7a95ad6ec564949"
62
+ "gitHead": "2ff891f214b3edfdad6e3b31f43f024ca026e019"
63
63
  }