From 538b24a26fe022bb48084a255c7b7642e0eb60a3 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Tue, 30 Apr 2024 10:13:21 +0200 Subject: [PATCH] Add popover capabilities to SelectNext listbox --- package.json | 1 + src/components/input/SelectNext.tsx | 164 ++++++++++++++---- src/components/input/test/SelectNext-test.js | 97 ++++++++++- .../patterns/prototype/SelectNextPage.tsx | 18 ++ yarn.lock | 28 ++- 5 files changed, 267 insertions(+), 41 deletions(-) diff --git a/package.json b/package.json index 6cb48378a..0c6bf2343 100644 --- a/package.json +++ b/package.json @@ -11,6 +11,7 @@ "@babel/preset-react": "^7.0.0", "@babel/preset-typescript": "^7.18.6", "@hypothesis/frontend-build": "^3.0.0", + "@hypothesis/frontend-testing": "^1.2.2", "@rollup/plugin-babel": "^6.0.0", "@rollup/plugin-commonjs": "^25.0.0", "@rollup/plugin-node-resolve": "^15.0.0", diff --git a/src/components/input/SelectNext.tsx b/src/components/input/SelectNext.tsx index 2cfb2127d..45dbf009e 100644 --- a/src/components/input/SelectNext.tsx +++ b/src/components/input/SelectNext.tsx @@ -5,6 +5,7 @@ import { useContext, useId, useLayoutEffect, + useMemo, useRef, useState, } from 'preact/hooks'; @@ -15,6 +16,7 @@ import { useFocusAway } from '../../hooks/use-focus-away'; import { useKeyPress } from '../../hooks/use-key-press'; import { useSyncedRef } from '../../hooks/use-synced-ref'; import type { CompositeProps } from '../../types'; +import { ListenerCollection } from '../../util/listener-collection'; import { downcastRef } from '../../util/typing'; import { MenuCollapseIcon, MenuExpandIcon } from '../icons'; import { inputGroupStyles } from './InputGroup'; @@ -103,38 +105,108 @@ function SelectOption({ SelectOption.displayName = 'SelectNext.Option'; -function useShouldDropUp( +const LISTBOX_TOGGLE_GAP = '.25rem'; + +type ListboxCSSProps = + | 'top' + | 'left' + | 'minWidth' + | 'marginBottom' + | 'bottom' + | 'marginTop'; + +function useListboxPositioning( buttonRef: RefObject, listboxRef: RefObject, listboxOpen: boolean, -): boolean { - const [shouldListboxDropUp, setShouldListboxDropUp] = useState(false); + asPopover: boolean, + right: boolean, +) { + const adjustListboxPositioning = useCallback(() => { + const listboxEl = listboxRef.current; + const buttonEl = buttonRef.current; - useLayoutEffect(() => { - // Reset shouldListboxDropUp so that it does not affect calculations next - // time listbox opens - if (!buttonRef.current || !listboxRef.current || !listboxOpen) { - setShouldListboxDropUp(false); - return; + if (!buttonEl || !listboxEl || !listboxOpen) { + return () => {}; } + /** + * Sets some CSS props in the listbox, and returns a cleanup function that + * resets them to their default values. + */ + const setListboxCSSProps = ( + props: Partial>, + ) => { + const entries = Object.entries(props) as Array<[ListboxCSSProps, string]>; + entries.forEach(([prop, value]) => (listboxEl.style[prop] = value)); + + return () => entries.map(([prop]) => (listboxEl.style[prop] = '')); + }; + const viewportHeight = window.innerHeight; - const { top: buttonDistanceToTop, bottom: buttonBottom } = - buttonRef.current.getBoundingClientRect(); + const { + top: buttonDistanceToTop, + bottom: buttonBottom, + left: buttonLeft, + height: buttonHeight, + width: buttonWidth, + } = buttonEl.getBoundingClientRect(); const buttonDistanceToBottom = viewportHeight - buttonBottom; - const { bottom: listboxBottom } = - listboxRef.current.getBoundingClientRect(); - const listboxDistanceToBottom = viewportHeight - listboxBottom; + const { height: listboxHeight, width: listboxWidth } = + listboxEl.getBoundingClientRect(); // The listbox should drop up only if there's not enough space below to - // fit it, and there's also more absolute space above than below - setShouldListboxDropUp( - listboxDistanceToBottom < 0 && - buttonDistanceToTop > buttonDistanceToBottom, - ); - }, [buttonRef, listboxRef, listboxOpen]); - - return shouldListboxDropUp; + // fit it, and also, there's more absolute space above than below + const shouldListboxDropUp = + buttonDistanceToBottom < listboxHeight && + buttonDistanceToTop > buttonDistanceToBottom; + + if (asPopover) { + const { top: bodyTop } = document.body.getBoundingClientRect(); + const absBodyTop = Math.abs(bodyTop); + + return setListboxCSSProps({ + minWidth: `${buttonWidth}px`, + top: shouldListboxDropUp + ? `calc(${absBodyTop + buttonDistanceToTop - listboxHeight}px - ${LISTBOX_TOGGLE_GAP})` + : `calc(${absBodyTop + buttonDistanceToTop + buttonHeight}px + ${LISTBOX_TOGGLE_GAP})`, + left: + right && listboxWidth > buttonWidth + ? `${buttonLeft - (listboxWidth - buttonWidth)}px` + : `${buttonLeft}px`, + }); + } + + // Set styles for non-popover mode + if (shouldListboxDropUp) { + return setListboxCSSProps({ + bottom: '100%', + marginBottom: LISTBOX_TOGGLE_GAP, + }); + } + + return setListboxCSSProps({ top: '100%', marginTop: LISTBOX_TOGGLE_GAP }); + }, [asPopover, buttonRef, listboxOpen, listboxRef, right]); + + useLayoutEffect(() => { + const cleanup = adjustListboxPositioning(); + + if (!asPopover) { + return cleanup; + } + + // Readjust listbox position when any element scrolls, just in case that + // affected the toggle button position. + const listeners = new ListenerCollection(); + listeners.add(document.body, 'scroll', adjustListboxPositioning, { + capture: true, + }); + + return () => { + cleanup(); + listeners.removeAll(); + }; + }, [adjustListboxPositioning, asPopover]); } export type SelectProps = CompositeProps & { @@ -166,6 +238,13 @@ export type SelectProps = CompositeProps & { 'aria-label'?: string; 'aria-labelledby'?: string; + + /** + * Test seam. + * Used to determine if current browser supports native popovers. + * Defaults to `'popover' in document.body` + */ + nativePopoverSupported?: boolean; }; function SelectMain({ @@ -182,18 +261,37 @@ function SelectMain({ right = false, 'aria-label': ariaLabel, 'aria-labelledby': ariaLabelledBy, + + /* istanbul ignore next - test seam */ + nativePopoverSupported = 'popover' in document.body, }: SelectProps) { - const [listboxOpen, setListboxOpen] = useState(false); - const closeListbox = useCallback(() => setListboxOpen(false), []); const wrapperRef = useRef(null); const listboxRef = useRef(null); + const [listboxOpen, setListboxOpen] = useState(false); + const toggleListbox = useCallback( + (open: boolean) => { + setListboxOpen(open); + if (nativePopoverSupported) { + listboxRef.current?.togglePopover(open); + } + }, + [nativePopoverSupported], + ); + const closeListbox = useCallback(() => toggleListbox(false), [toggleListbox]); const listboxId = useId(); const buttonRef = useSyncedRef(elementRef); const defaultButtonId = useId(); - const shouldListboxDropUp = useShouldDropUp( + const extraProps = useMemo( + () => (nativePopoverSupported ? { popover: true } : {}), + [nativePopoverSupported], + ); + + useListboxPositioning( buttonRef, listboxRef, listboxOpen, + nativePopoverSupported, + right, ); const selectValue = useCallback( @@ -257,11 +355,11 @@ function SelectMain({ aria-label={ariaLabel} aria-labelledby={ariaLabelledBy} ref={downcastRef(buttonRef)} - onClick={() => setListboxOpen(prev => !prev)} + onClick={() => toggleListbox(!listboxOpen)} onKeyDown={e => { if (e.key === 'ArrowDown' && !listboxOpen) { e.preventDefault(); - setListboxOpen(true); + toggleListbox(true); } }} data-testid="select-toggle-button" @@ -273,18 +371,17 @@ function SelectMain({
    ({ aria-labelledby={buttonId ?? defaultButtonId} aria-orientation="vertical" data-testid="select-listbox" + data-listbox-open={listboxOpen} > {children}
diff --git a/src/components/input/test/SelectNext-test.js b/src/components/input/test/SelectNext-test.js index 8a49bab51..d4f6a7fff 100644 --- a/src/components/input/test/SelectNext-test.js +++ b/src/components/input/test/SelectNext-test.js @@ -1,3 +1,4 @@ +import { waitFor } from '@hypothesis/frontend-testing'; import { mount } from 'enzyme'; import { checkAccessibility } from '../../../../test/util/accessibility.js'; @@ -28,6 +29,8 @@ describe('SelectNext', () => { container.style.paddingTop = `${paddingTop}px`; document.body.append(container); + props.nativePopoverSupported = props.nativePopoverSupported ?? false; + const wrapper = mount( {items.map(item => ( @@ -74,10 +77,18 @@ describe('SelectNext', () => { const getListbox = wrapper => wrapper.find('[data-testid="select-listbox"]'); const isListboxClosed = wrapper => - getListbox(wrapper).prop('className').includes('hidden'); + getListbox(wrapper).prop('data-listbox-open') === false; + + const listboxDidDropUp = wrapper => { + const { top: listboxTop } = getListbox(wrapper) + .getDOMNode() + .getBoundingClientRect(); + const { top: buttonTop } = getToggleButton(wrapper) + .getDOMNode() + .getBoundingClientRect(); - const listboxDidDropUp = wrapper => - getListbox(wrapper).prop('className').includes('bottom-full'); + return listboxTop < buttonTop; + }; it('changes selected value when an option is clicked', () => { const onChange = sinon.stub(); @@ -261,11 +272,32 @@ describe('SelectNext', () => { }); [ - { containerPaddingTop: 0, shouldDropUp: false }, - { containerPaddingTop: 1000, shouldDropUp: true }, - ].forEach(({ containerPaddingTop, shouldDropUp }) => { + { + containerPaddingTop: 0, + shouldDropUp: false, + nativePopoverSupported: true, + }, + { + containerPaddingTop: 0, + shouldDropUp: false, + nativePopoverSupported: false, + }, + { + containerPaddingTop: 1000, + shouldDropUp: true, + nativePopoverSupported: true, + }, + { + containerPaddingTop: 1000, + shouldDropUp: true, + nativePopoverSupported: false, + }, + ].forEach(({ containerPaddingTop, shouldDropUp, nativePopoverSupported }) => { it('makes listbox drop up or down based on available space below', () => { - const wrapper = createComponent({}, { paddingTop: containerPaddingTop }); + const wrapper = createComponent( + { nativePopoverSupported }, + { paddingTop: containerPaddingTop }, + ); toggleListbox(wrapper); assert.equal(listboxDidDropUp(wrapper), shouldDropUp); @@ -282,6 +314,57 @@ describe('SelectNext', () => { }); }); + context('when popover is supported', () => { + it('opens listbox via popover API', async () => { + const wrapper = createComponent({ nativePopoverSupported: true }); + let resolve; + const promise = new Promise(res => (resolve = res)); + + getListbox(wrapper).getDOMNode().addEventListener('toggle', resolve); + toggleListbox(wrapper); + + // This test will timeout if the toggle event is not dispatched + await promise; + }); + }); + + context('when listbox is bigger than toggle button', () => { + [ + { + nativePopoverSupported: true, + getListboxLeft: wrapper => { + const leftStyle = getListbox(wrapper).getDOMNode().style.left; + // Remove last 2 chars, which are the `px` unit indicator + return Number(leftStyle.slice(0, -2)); + }, + }, + { + nativePopoverSupported: false, + getListboxLeft: wrapper => + getListbox(wrapper).getDOMNode().getBoundingClientRect().left, + }, + ].forEach(({ nativePopoverSupported, getListboxLeft }) => { + it('aligns listbox to the right if `right` prop is true', async () => { + const wrapper = createComponent({ + nativePopoverSupported, + right: true, + buttonClasses: '!w-8', // Set a small width in the button + }); + toggleListbox(wrapper); + + // Wait for listbox to be open + await waitFor(() => !isListboxClosed(wrapper)); + + const { left: buttonLeft } = getToggleButton(wrapper) + .getDOMNode() + .getBoundingClientRect(); + const listboxLeft = getListboxLeft(wrapper); + + assert.isTrue(listboxLeft < buttonLeft); + }); + }); + }); + it( 'should pass a11y checks', checkAccessibility([ diff --git a/src/pattern-library/components/patterns/prototype/SelectNextPage.tsx b/src/pattern-library/components/patterns/prototype/SelectNextPage.tsx index 4ba0b0829..225cbd471 100644 --- a/src/pattern-library/components/patterns/prototype/SelectNextPage.tsx +++ b/src/pattern-library/components/patterns/prototype/SelectNextPage.tsx @@ -1,6 +1,7 @@ import classnames from 'classnames'; import { useCallback, useId, useMemo, useState } from 'preact/hooks'; +import { Link } from '../../../..'; import { ArrowLeftIcon, ArrowRightIcon } from '../../../../components/icons'; import type { SelectNextProps } from '../../../../components/input'; import { IconButton, InputGroup } from '../../../../components/input'; @@ -206,6 +207,23 @@ export default function SelectNextPage() { SelectNext toggles a listbox where Options {"'"} UI can be customized and values can be objects.

+

+ In browsers that support it, the listbox uses the{' '} + + popover + {' '} + attribute and gets toggled via{' '} + + popover API + + . Otherwise, it is rendered as an absolute-positioned element. +

diff --git a/yarn.lock b/yarn.lock index 88ee40b94..90b41bf90 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2060,6 +2060,7 @@ __metadata: "@babel/preset-react": ^7.0.0 "@babel/preset-typescript": ^7.18.6 "@hypothesis/frontend-build": ^3.0.0 + "@hypothesis/frontend-testing": ^1.2.2 "@rollup/plugin-babel": ^6.0.0 "@rollup/plugin-commonjs": ^25.0.0 "@rollup/plugin-node-resolve": ^15.0.0 @@ -2112,6 +2113,17 @@ __metadata: languageName: unknown linkType: soft +"@hypothesis/frontend-testing@npm:^1.2.2": + version: 1.2.2 + resolution: "@hypothesis/frontend-testing@npm:1.2.2" + dependencies: + axe-core: ^4.8.2 + enzyme: ^3.11.0 + preact: ^10.18.1 + checksum: b3892f6f1db87387aa60a420f0d2a7c89c5031a1375273cce6777aba335f5d01b223a5fee499f1fbf418044eea6c85684d8488647e8b353d9637c05ba29d15de + languageName: node + linkType: hard + "@isaacs/cliui@npm:^8.0.2": version: 8.0.2 resolution: "@isaacs/cliui@npm:8.0.2" @@ -3335,6 +3347,13 @@ __metadata: languageName: node linkType: hard +"axe-core@npm:^4.8.2": + version: 4.9.1 + resolution: "axe-core@npm:4.9.1" + checksum: 41d9227871781f96c2952e2a777fca73624959dd0e98864f6d82806a77602f82b4fc490852082a7e524d8cd864e50d8b4d9931819b4a150112981d8c932110c5 + languageName: node + linkType: hard + "axobject-query@npm:^3.2.1": version: 3.2.1 resolution: "axobject-query@npm:3.2.1" @@ -4815,7 +4834,7 @@ __metadata: languageName: node linkType: hard -"enzyme@npm:^3.8.0": +"enzyme@npm:^3.11.0, enzyme@npm:^3.8.0": version: 3.11.0 resolution: "enzyme@npm:3.11.0" dependencies: @@ -9336,6 +9355,13 @@ __metadata: languageName: node linkType: hard +"preact@npm:^10.18.1": + version: 10.21.0 + resolution: "preact@npm:10.21.0" + checksum: 04acdd300e779eec230256d043120e564ac76bb535b05a9b23949f8641bf8e135098d0bd12c008cc4316174469356d09a2326dde6acbd8f539d1b4bf2231cbaa + languageName: node + linkType: hard + "preact@npm:^10.5.6": version: 10.20.2 resolution: "preact@npm:10.20.2"