@codecademy/gamut 68.2.3-alpha.ea023d.0 → 68.2.3-alpha.f19d29.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/DataList/Controls/FilterControl.js +1 -1
- package/dist/DatePicker/Calendar/CalendarBody.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarBody.js +145 -0
- package/dist/DatePicker/Calendar/CalendarFooter.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarFooter.js +57 -0
- package/dist/DatePicker/Calendar/CalendarHeader.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarHeader.js +46 -0
- package/dist/DatePicker/Calendar/CalendarNavLastMonth.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarNavLastMonth.js +30 -0
- package/dist/DatePicker/Calendar/CalendarNavNextMonth.d.ts +3 -0
- package/dist/DatePicker/Calendar/CalendarNavNextMonth.js +30 -0
- package/dist/DatePicker/Calendar/CalendarWrapper.d.ts +8 -0
- package/dist/DatePicker/Calendar/CalendarWrapper.js +27 -0
- package/dist/DatePicker/Calendar/index.d.ts +6 -0
- package/dist/DatePicker/Calendar/index.js +6 -0
- package/dist/DatePicker/Calendar/types.d.ts +77 -0
- package/dist/DatePicker/Calendar/types.js +1 -0
- package/dist/DatePicker/Calendar/utils/dateGrid.d.ts +38 -0
- package/dist/DatePicker/Calendar/utils/dateGrid.js +103 -0
- package/dist/DatePicker/Calendar/utils/elements.d.ts +18 -0
- package/dist/DatePicker/Calendar/utils/elements.js +116 -0
- package/dist/DatePicker/Calendar/utils/format.d.ts +28 -0
- package/dist/DatePicker/Calendar/utils/format.js +72 -0
- package/dist/DatePicker/Calendar/utils/keyHandler.d.ts +12 -0
- package/dist/DatePicker/Calendar/utils/keyHandler.js +126 -0
- package/dist/DatePicker/DatePicker.d.ts +7 -0
- package/dist/DatePicker/DatePicker.js +157 -0
- package/dist/DatePicker/DatePickerCalendar.d.ts +11 -0
- package/dist/DatePicker/DatePickerCalendar.js +168 -0
- package/dist/DatePicker/DatePickerContext.d.ts +10 -0
- package/dist/DatePicker/DatePickerContext.js +18 -0
- package/dist/DatePicker/DatePickerInput/Segment/DatePickerInputSegment.d.ts +23 -0
- package/dist/DatePicker/DatePickerInput/Segment/DatePickerInputSegment.js +131 -0
- package/dist/DatePicker/DatePickerInput/Segment/elements.d.ts +16 -0
- package/dist/DatePicker/DatePickerInput/Segment/elements.js +36 -0
- package/dist/DatePicker/DatePickerInput/Segment/index.d.ts +4 -0
- package/dist/DatePicker/DatePickerInput/Segment/index.js +3 -0
- package/dist/DatePicker/DatePickerInput/Segment/segmentUtils.d.ts +50 -0
- package/dist/DatePicker/DatePickerInput/Segment/segmentUtils.js +201 -0
- package/dist/DatePicker/DatePickerInput/elements.d.ts +15 -0
- package/dist/DatePicker/DatePickerInput/elements.js +19 -0
- package/dist/DatePicker/DatePickerInput/index.d.ts +13 -0
- package/dist/DatePicker/DatePickerInput/index.js +206 -0
- package/dist/DatePicker/DatePickerInput/utils.d.ts +17 -0
- package/dist/DatePicker/DatePickerInput/utils.js +50 -0
- package/dist/DatePicker/index.d.ts +5 -0
- package/dist/DatePicker/index.js +5 -0
- package/dist/DatePicker/types.d.ts +86 -0
- package/dist/DatePicker/types.js +1 -0
- package/dist/DatePicker/utils/dateSelect.d.ts +29 -0
- package/dist/DatePicker/utils/dateSelect.js +145 -0
- package/dist/DatePicker/utils/locale.d.ts +38 -0
- package/dist/DatePicker/utils/locale.js +93 -0
- package/dist/DatePicker/utils/translations.d.ts +15 -0
- package/dist/DatePicker/utils/translations.js +10 -0
- package/dist/FocusTrap/index.d.ts +2 -2
- package/dist/Pagination/index.js +5 -10
- package/dist/PopoverContainer/PopoverContainer.js +23 -7
- package/dist/PopoverContainer/types.d.ts +7 -1
- package/dist/index.d.ts +2 -0
- package/dist/index.js +2 -0
- package/package.json +7 -6
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
// Replaces `Intl.Locale` when missing or incomplete (e.g. no `getWeekInfo` in Firefox).
|
|
2
|
+
// https://formatjs.github.io/docs/polyfills/intl-locale/
|
|
3
|
+
import '@formatjs/intl-locale/polyfill.js';
|
|
4
|
+
import { useEffect, useMemo, useState } from 'react';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* The runtime default locale string (user agent), matching what `Intl` uses when no locale is passed.
|
|
8
|
+
*/
|
|
9
|
+
export const getDefaultLocaleTag = () => new Intl.DateTimeFormat().resolvedOptions().locale;
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Resolves `Intl.LocalesArgument` (or `undefined`) to a stable `Intl.Locale` instance for formatting
|
|
13
|
+
* and locale metadata (e.g. `getWeekInfo()` in supporting environments).
|
|
14
|
+
*
|
|
15
|
+
* - `undefined` → default runtime locale via {@link getDefaultLocaleTag}
|
|
16
|
+
* - `Intl.Locale` → returned as-is (no duplicate allocation)
|
|
17
|
+
* - `string` / `readonly string[]` → `new Intl.Locale(...)`
|
|
18
|
+
*/
|
|
19
|
+
export const resolveLocale = locales => {
|
|
20
|
+
if (locales === undefined) {
|
|
21
|
+
return new Intl.Locale(getDefaultLocaleTag());
|
|
22
|
+
}
|
|
23
|
+
if (locales instanceof Intl.Locale) {
|
|
24
|
+
return locales;
|
|
25
|
+
}
|
|
26
|
+
if (typeof locales === 'string') {
|
|
27
|
+
return new Intl.Locale(locales);
|
|
28
|
+
}
|
|
29
|
+
const first = locales[0];
|
|
30
|
+
if (first === undefined) {
|
|
31
|
+
return new Intl.Locale(getDefaultLocaleTag());
|
|
32
|
+
}
|
|
33
|
+
if (typeof first === 'string') {
|
|
34
|
+
return new Intl.Locale(first);
|
|
35
|
+
}
|
|
36
|
+
return first instanceof Intl.Locale ? first : new Intl.Locale(String(first));
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Memoized {@link resolveLocale} for calendar subcomponents. Pass the same `locale` prop you accept
|
|
41
|
+
* from `CalendarBaseProps` (optional `Intl.LocalesArgument`).
|
|
42
|
+
*/
|
|
43
|
+
export const useResolvedLocale = locale => useMemo(() => resolveLocale(locale), [locale]);
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Convert an Intl.Locale to a string. This is necessary the Intl.DateTimeFormat constructor only accepts a string in some versions of TS.
|
|
47
|
+
* @param locale - The Intl.Locale to convert to a string.
|
|
48
|
+
* @returns The stringified locale.
|
|
49
|
+
*/
|
|
50
|
+
export const stringifyLocale = locale => locale.toString();
|
|
51
|
+
|
|
52
|
+
/** ISO weekday: 1 = Monday … 7 = Sunday (matches `Intl.Locale#getWeekInfo().firstDay`). */
|
|
53
|
+
|
|
54
|
+
/** `getWeekInfo` is stage-3; typings may lag behind runtime / polyfill. */
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* First calendar column weekday from `locale` (via `getWeekInfo()`), or explicit override.
|
|
58
|
+
* - `weekStartsOnOverride` — ISO weekday **1–7** (Monday … Sunday), same as `getWeekInfo().firstDay`
|
|
59
|
+
* - omitted → `locale.getWeekInfo().firstDay` when available, else **7** (Sunday)
|
|
60
|
+
*/
|
|
61
|
+
export const getIsoFirstDayFromLocale = (locale, weekStartsOnOverride) => {
|
|
62
|
+
if (weekStartsOnOverride) return weekStartsOnOverride;
|
|
63
|
+
try {
|
|
64
|
+
const getWeekInfo = locale.getWeekInfo?.bind(locale);
|
|
65
|
+
if (typeof getWeekInfo === 'function') {
|
|
66
|
+
const {
|
|
67
|
+
firstDay
|
|
68
|
+
} = getWeekInfo();
|
|
69
|
+
if (typeof firstDay === 'number' && firstDay >= 1 && firstDay <= 7) {
|
|
70
|
+
return firstDay;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
} catch {
|
|
74
|
+
/* ignore */
|
|
75
|
+
}
|
|
76
|
+
return 7;
|
|
77
|
+
};
|
|
78
|
+
|
|
79
|
+
/**
|
|
80
|
+
* Hook: resolved first weekday for the calendar grid. Re-reads after mount so async polyfills
|
|
81
|
+
* (e.g. Firefox) can install `getWeekInfo` before the first paint in some bundles.
|
|
82
|
+
*/
|
|
83
|
+
export const useIsoFirstWeekday = (locale, weekStartsOnOverride) => {
|
|
84
|
+
const [firstDay, setFirstDay] = useState(() => getIsoFirstDayFromLocale(locale, weekStartsOnOverride));
|
|
85
|
+
useEffect(() => {
|
|
86
|
+
setFirstDay(getIsoFirstDayFromLocale(locale, weekStartsOnOverride));
|
|
87
|
+
const t = setTimeout(() => {
|
|
88
|
+
setFirstDay(getIsoFirstDayFromLocale(locale, weekStartsOnOverride));
|
|
89
|
+
}, 0);
|
|
90
|
+
return () => clearTimeout(t);
|
|
91
|
+
}, [locale, weekStartsOnOverride]);
|
|
92
|
+
return firstDay;
|
|
93
|
+
};
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/** Optional translations for DatePicker UI strings. Pass to override defaults. */
|
|
2
|
+
export interface DatePickerTranslations {
|
|
3
|
+
/** Label for the clear date button (default: "Clear"). */
|
|
4
|
+
clearText?: string;
|
|
5
|
+
/** Default label for the date input in single mode (default: "Date"). */
|
|
6
|
+
dateLabel?: string;
|
|
7
|
+
/** Default label for the start date input in range mode (default: "Start date"). */
|
|
8
|
+
startDateLabel?: string;
|
|
9
|
+
/** Default label for the end date input in range mode (default: "End date"). */
|
|
10
|
+
endDateLabel?: string;
|
|
11
|
+
/** aria-label for the calendar dialog (default: "Choose date"). */
|
|
12
|
+
calendarDialogAriaLabel?: string;
|
|
13
|
+
}
|
|
14
|
+
/** Default UI strings; pass translations prop to override. */
|
|
15
|
+
export declare const DEFAULT_DATE_PICKER_TRANSLATIONS: Required<DatePickerTranslations>;
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/** Optional translations for DatePicker UI strings. Pass to override defaults. */
|
|
2
|
+
|
|
3
|
+
/** Default UI strings; pass translations prop to override. */
|
|
4
|
+
export const DEFAULT_DATE_PICKER_TRANSLATIONS = {
|
|
5
|
+
clearText: 'Clear',
|
|
6
|
+
dateLabel: 'Date',
|
|
7
|
+
startDateLabel: 'Start date',
|
|
8
|
+
endDateLabel: 'End date',
|
|
9
|
+
calendarDialogAriaLabel: 'Choose date'
|
|
10
|
+
};
|
|
@@ -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/Pagination/index.js
CHANGED
|
@@ -1,8 +1,7 @@
|
|
|
1
1
|
import { MiniChevronLeftIcon, MiniChevronRightIcon } from '@codecademy/gamut-icons';
|
|
2
|
-
import {
|
|
3
|
-
import { useMemo, useRef, useState } from 'react';
|
|
2
|
+
import { useMemo, useState } from 'react';
|
|
4
3
|
import * as React from 'react';
|
|
5
|
-
import {
|
|
4
|
+
import { HiddenText } from '..';
|
|
6
5
|
import { FlexBox } from '../Box';
|
|
7
6
|
import { AnimatedFadeButton, AnimatedSlideButton } from './AnimatedPaginationButtons';
|
|
8
7
|
import { EllipsisButton } from './EllipsisButton';
|
|
@@ -24,8 +23,6 @@ export const Pagination = ({
|
|
|
24
23
|
const [currentPage, setCurrentPage] = useState(pageNumber ?? defaultPageNumber);
|
|
25
24
|
const [liveText, setLiveText] = useState('');
|
|
26
25
|
const [shownPageArray, setShownPageArray] = useState([0]);
|
|
27
|
-
const rootRef = useRef(null);
|
|
28
|
-
const isRtl = useElementDir(rootRef) === 'rtl';
|
|
29
26
|
const showSkipToButtons = !!(type === undefined && totalPages >= 10 || type === 'includeSkipToButtons');
|
|
30
27
|
const changeShownPages = shouldPagesChange({
|
|
31
28
|
chapterSize,
|
|
@@ -70,16 +67,14 @@ export const Pagination = ({
|
|
|
70
67
|
chapterSize
|
|
71
68
|
}) : 'initial'}`
|
|
72
69
|
},
|
|
73
|
-
|
|
74
|
-
children: [/*#__PURE__*/_jsx(Text, {
|
|
70
|
+
children: [/*#__PURE__*/_jsx(HiddenText, {
|
|
75
71
|
"aria-live": "polite",
|
|
76
|
-
screenreader: true,
|
|
77
72
|
children: liveText
|
|
78
73
|
}), /*#__PURE__*/_jsx(AnimatedFadeButton, {
|
|
79
74
|
"aria-label": `Navigate back to page ${currentPage - 1}`,
|
|
80
75
|
buttonType: variant,
|
|
81
76
|
href: navigation,
|
|
82
|
-
icon:
|
|
77
|
+
icon: MiniChevronLeftIcon,
|
|
83
78
|
showButton: currentPage === 1 ? 'hidden' : 'shown',
|
|
84
79
|
onClick: () => changeHandler(currentPage - 1)
|
|
85
80
|
}), showSkipToButtons && /*#__PURE__*/_jsxs(_Fragment, {
|
|
@@ -134,7 +129,7 @@ export const Pagination = ({
|
|
|
134
129
|
"aria-label": `Navigate forward to page ${currentPage + 1}`,
|
|
135
130
|
buttonType: variant,
|
|
136
131
|
href: navigation,
|
|
137
|
-
icon:
|
|
132
|
+
icon: MiniChevronRightIcon,
|
|
138
133
|
showButton: currentPage === totalPages ? 'hidden' : 'shown',
|
|
139
134
|
onClick: () => changeHandler(currentPage + 1)
|
|
140
135
|
})]
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import _styled from "@emotion/styled/base";
|
|
2
|
-
import { system
|
|
2
|
+
import { system } from '@codecademy/gamut-styles';
|
|
3
3
|
import { variance } from '@codecademy/variance';
|
|
4
4
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
|
5
5
|
import * as React from 'react';
|
|
@@ -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":"AAqBuB","file":"../../src/PopoverContainer/PopoverContainer.tsx","sourcesContent":["import {\n  system,\n  useElementDir,\n  useLogicalProperties,\n} 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(targetRef);\n\n  // Keep onRequestClose ref up to date\n  useEffect(() => {\n    onRequestCloseRef.current = onRequestClose;\n  }, [onRequestClose]);\n\n  const isRtl = useElementDir(targetRef) === 'rtl';\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  // Log logical properties to the console TEST CODE\n  const logicalProperties = useLogicalProperties();\n  console.log('dir', isRtl, 'logicalProperties', logicalProperties);\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);
|
|
@@ -53,7 +54,25 @@ export const PopoverContainer = ({
|
|
|
53
54
|
useEffect(() => {
|
|
54
55
|
onRequestCloseRef.current = onRequestClose;
|
|
55
56
|
}, [onRequestClose]);
|
|
56
|
-
|
|
57
|
+
|
|
58
|
+
// Detect RTL direction from the target element and watch for attribute changes so the
|
|
59
|
+
// position recalculates when changes occur
|
|
60
|
+
const [isRtl, setIsRtl] = useState(false);
|
|
61
|
+
useEffect(() => {
|
|
62
|
+
const checkDirection = () => {
|
|
63
|
+
const target = targetRef?.current;
|
|
64
|
+
const el = target instanceof Element ? target : document.documentElement;
|
|
65
|
+
setIsRtl(getComputedStyle(el).direction === 'rtl');
|
|
66
|
+
};
|
|
67
|
+
checkDirection();
|
|
68
|
+
const observer = new MutationObserver(checkDirection);
|
|
69
|
+
observer.observe(document.documentElement, {
|
|
70
|
+
attributes: true,
|
|
71
|
+
attributeFilter: ['dir'],
|
|
72
|
+
subtree: true
|
|
73
|
+
});
|
|
74
|
+
return () => observer.disconnect();
|
|
75
|
+
}, [targetRef]);
|
|
57
76
|
const popoverPosition = useMemo(() => {
|
|
58
77
|
if (parent !== undefined) {
|
|
59
78
|
return getPosition({
|
|
@@ -71,10 +90,6 @@ export const PopoverContainer = ({
|
|
|
71
90
|
physicalStyles: undefined
|
|
72
91
|
};
|
|
73
92
|
}, [parent, x, y, offset, alignment, invertAxis, isRtl]);
|
|
74
|
-
|
|
75
|
-
// Log logical properties to the console TEST CODE
|
|
76
|
-
const logicalProperties = useLogicalProperties();
|
|
77
|
-
console.log('dir', isRtl, 'logicalProperties', logicalProperties);
|
|
78
93
|
useEffect(() => {
|
|
79
94
|
const target = targetRef?.current;
|
|
80
95
|
if (!target) return;
|
|
@@ -175,6 +190,7 @@ export const PopoverContainer = ({
|
|
|
175
190
|
if (!isOpen || !targetRef) return null;
|
|
176
191
|
const content = /*#__PURE__*/_jsx(FocusTrap, {
|
|
177
192
|
allowPageInteraction: inline || allowPageInteraction,
|
|
193
|
+
focusOnProps: focusOnProps,
|
|
178
194
|
onClickOutside: handleClickOutside,
|
|
179
195
|
onEscapeKey: onRequestClose,
|
|
180
196
|
children: /*#__PURE__*/_jsx(PopoverContent, {
|
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import { RefObject } from 'react';
|
|
2
2
|
import { WithChildrenProp } from '../utils';
|
|
3
3
|
export type Alignments = 'top-left' | 'top-right' | 'bottom-left' | 'bottom-right' | 'top' | 'bottom' | 'left' | 'right';
|
|
4
|
-
export
|
|
4
|
+
export interface TargetRef extends Pick<HTMLDivElement, 'getBoundingClientRect' | 'contains' | 'offsetHeight' | 'offsetWidth' | 'offsetTop' | 'offsetLeft' | 'offsetParent'> {
|
|
5
|
+
}
|
|
5
6
|
export interface PositionContext {
|
|
6
7
|
width: number;
|
|
7
8
|
height: number;
|
|
@@ -64,4 +65,9 @@ export interface PopoverContainerProps extends PopoverAlignment, WithChildrenPro
|
|
|
64
65
|
* Defaults to false.
|
|
65
66
|
*/
|
|
66
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']>;
|
|
67
73
|
}
|
package/dist/index.d.ts
CHANGED
|
@@ -16,6 +16,8 @@ export * from './Card';
|
|
|
16
16
|
export * from './Coachmark';
|
|
17
17
|
export * from './ConnectedForm';
|
|
18
18
|
export * from './ContentContainer';
|
|
19
|
+
export * from './DatePicker';
|
|
20
|
+
export * from './DatePicker/Calendar';
|
|
19
21
|
export * from './DelayedRenderWrapper';
|
|
20
22
|
export * from './Disclosure';
|
|
21
23
|
export * from './DataList';
|
package/dist/index.js
CHANGED
|
@@ -15,6 +15,8 @@ export * from './Card';
|
|
|
15
15
|
export * from './Coachmark';
|
|
16
16
|
export * from './ConnectedForm';
|
|
17
17
|
export * from './ContentContainer';
|
|
18
|
+
export * from './DatePicker';
|
|
19
|
+
export * from './DatePicker/Calendar';
|
|
18
20
|
export * from './DelayedRenderWrapper';
|
|
19
21
|
export * from './Disclosure';
|
|
20
22
|
export * from './DataList';
|
package/package.json
CHANGED
|
@@ -1,14 +1,15 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@codecademy/gamut",
|
|
3
3
|
"description": "Styleguide & Component library for Codecademy",
|
|
4
|
-
"version": "68.2.3-alpha.
|
|
4
|
+
"version": "68.2.3-alpha.f19d29.0",
|
|
5
5
|
"author": "Codecademy Engineering <dev@codecademy.com>",
|
|
6
6
|
"dependencies": {
|
|
7
|
-
"@codecademy/gamut-icons": "9.57.3-alpha.
|
|
8
|
-
"@codecademy/gamut-illustrations": "0.58.10-alpha.
|
|
9
|
-
"@codecademy/gamut-patterns": "0.10.29-alpha.
|
|
10
|
-
"@codecademy/gamut-styles": "17.13.2-alpha.
|
|
11
|
-
"@codecademy/variance": "0.26.2-alpha.
|
|
7
|
+
"@codecademy/gamut-icons": "9.57.3-alpha.f19d29.0",
|
|
8
|
+
"@codecademy/gamut-illustrations": "0.58.10-alpha.f19d29.0",
|
|
9
|
+
"@codecademy/gamut-patterns": "0.10.29-alpha.f19d29.0",
|
|
10
|
+
"@codecademy/gamut-styles": "17.13.2-alpha.f19d29.0",
|
|
11
|
+
"@codecademy/variance": "0.26.2-alpha.f19d29.0",
|
|
12
|
+
"@formatjs/intl-locale": "^5.3.1",
|
|
12
13
|
"@react-aria/interactions": "3.25.0",
|
|
13
14
|
"@types/marked": "^4.0.8",
|
|
14
15
|
"@vidstack/react": "^1.12.12",
|