@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.
- package/dist/ConnectedForm/utils.d.ts +1 -1
- package/dist/DatePicker/Calendar/Calendar.d.ts +9 -0
- package/dist/DatePicker/Calendar/Calendar.js +28 -0
- package/dist/DatePicker/Calendar/CalendarBody.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarBody.js +155 -0
- package/dist/DatePicker/Calendar/CalendarFooter.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarFooter.js +54 -0
- package/dist/DatePicker/Calendar/CalendarHeader.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarHeader.js +67 -0
- package/dist/DatePicker/Calendar/index.d.ts +6 -0
- package/dist/DatePicker/Calendar/index.js +5 -0
- package/dist/DatePicker/Calendar/types.d.ts +60 -0
- package/dist/DatePicker/Calendar/types.js +1 -0
- package/dist/DatePicker/Calendar/utils/dateGrid.d.ts +30 -0
- package/dist/DatePicker/Calendar/utils/dateGrid.js +93 -0
- package/dist/DatePicker/Calendar/utils/format.d.ts +61 -0
- package/dist/DatePicker/Calendar/utils/format.js +184 -0
- package/dist/DatePicker/Calendar/utils/index.d.ts +3 -0
- package/dist/DatePicker/Calendar/utils/index.js +3 -0
- package/dist/DatePicker/Calendar/utils/keyHandler.d.ts +13 -0
- package/dist/DatePicker/Calendar/utils/keyHandler.js +116 -0
- package/dist/DatePicker/Calendar/utils/validation.d.ts +13 -0
- package/dist/DatePicker/Calendar/utils/validation.js +23 -0
- package/dist/DatePicker/DatePicker.d.ts +8 -0
- package/dist/DatePicker/DatePicker.js +147 -0
- package/dist/DatePicker/DatePickerCalendar.d.ts +13 -0
- package/dist/DatePicker/DatePickerCalendar.js +130 -0
- package/dist/DatePicker/DatePickerContext.d.ts +11 -0
- package/dist/DatePicker/DatePickerContext.js +18 -0
- package/dist/DatePicker/DatePickerInput.d.ts +16 -0
- package/dist/DatePicker/DatePickerInput.js +135 -0
- package/dist/DatePicker/index.d.ts +13 -0
- package/dist/DatePicker/index.js +10 -0
- package/dist/DatePicker/translations.d.ts +3 -0
- package/dist/DatePicker/translations.js +4 -0
- package/dist/DatePicker/types.d.ts +85 -0
- package/dist/DatePicker/types.js +1 -0
- package/dist/DatePicker/utils.d.ts +3 -0
- package/dist/DatePicker/utils.js +71 -0
- package/dist/FocusTrap/index.d.ts +2 -2
- package/dist/List/elements.d.ts +1 -1
- package/dist/PopoverContainer/PopoverContainer.js +3 -1
- package/dist/PopoverContainer/types.d.ts +5 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +1 -0
- 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,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>;
|
package/dist/List/elements.d.ts
CHANGED
|
@@ -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" | "
|
|
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.
|
|
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": "
|
|
62
|
+
"gitHead": "2ff891f214b3edfdad6e3b31f43f024ca026e019"
|
|
63
63
|
}
|