UNPKG

@base-ui-components/react

Version:

Base UI is a library of headless ('unstyled') React components and low-level hooks. You gain complete control over your app's CSS and accessibility features.

210 lines (207 loc) 7.21 kB
'use client'; import * as React from 'react'; import { useIsoLayoutEffect } from '@base-ui-components/utils/useIsoLayoutEffect'; import { useValueAsRef } from '@base-ui-components/utils/useValueAsRef'; import { isMouseWithinBounds } from '@base-ui-components/utils/isMouseWithinBounds'; import { useTimeout } from '@base-ui-components/utils/useTimeout'; import { useStore } from '@base-ui-components/utils/store'; import { useSelectRootContext } from "../root/SelectRootContext.js"; import { useCompositeListItem, IndexGuessBehavior } from "../../composite/list/useCompositeListItem.js"; import { useRenderElement } from "../../utils/useRenderElement.js"; import { SelectItemContext } from "./SelectItemContext.js"; import { selectors } from "../store.js"; import { useButton } from "../../use-button/index.js"; import { createChangeEventDetails } from "../../utils/createBaseUIEventDetails.js"; import { REASONS } from "../../utils/reasons.js"; import { compareItemEquality, removeItem } from "../../utils/itemEquality.js"; /** * An individual option in the select popup. * Renders a `<div>` element. * * Documentation: [Base UI Select](https://base-ui.com/react/components/select) */ import { jsx as _jsx } from "react/jsx-runtime"; export const SelectItem = /*#__PURE__*/React.memo(/*#__PURE__*/React.forwardRef(function SelectItem(componentProps, forwardedRef) { const { render, className, value = null, label, disabled = false, nativeButton = false, ...elementProps } = componentProps; const textRef = React.useRef(null); const listItem = useCompositeListItem({ label, textRef, indexGuessBehavior: IndexGuessBehavior.GuessFromOrder }); const { store, getItemProps, setOpen, setValue, selectionRef, typingRef, valuesRef, keyboardActiveRef, multiple } = useSelectRootContext(); const highlightTimeout = useTimeout(); const highlighted = useStore(store, selectors.isActive, listItem.index); const selected = useStore(store, selectors.isSelected, listItem.index, value); const selectedByFocus = useStore(store, selectors.isSelectedByFocus, listItem.index); const isItemEqualToValue = useStore(store, selectors.isItemEqualToValue); const index = listItem.index; const hasRegistered = index !== -1; const itemRef = React.useRef(null); const indexRef = useValueAsRef(index); useIsoLayoutEffect(() => { if (!hasRegistered) { return undefined; } const values = valuesRef.current; values[index] = value; return () => { delete values[index]; }; }, [hasRegistered, index, value, valuesRef]); useIsoLayoutEffect(() => { if (!hasRegistered) { return undefined; } const selectedValue = store.state.value; let candidate = selectedValue; if (multiple && Array.isArray(selectedValue) && selectedValue.length > 0) { candidate = selectedValue[selectedValue.length - 1]; } if (candidate !== undefined && compareItemEquality(candidate, value, isItemEqualToValue)) { store.set('selectedIndex', index); } return undefined; }, [hasRegistered, index, multiple, isItemEqualToValue, store, value]); const state = React.useMemo(() => ({ disabled, selected, highlighted }), [disabled, selected, highlighted]); const rootProps = getItemProps({ active: highlighted, selected }); // With our custom `focusItemOnHover` implementation, this interferes with the logic and can // cause the index state to be stuck when leaving the select popup. rootProps.onFocus = undefined; rootProps.id = undefined; const lastKeyRef = React.useRef(null); const pointerTypeRef = React.useRef('mouse'); const didPointerDownRef = React.useRef(false); const { getButtonProps, buttonRef } = useButton({ disabled, focusableWhenDisabled: true, native: nativeButton }); function commitSelection(event) { const selectedValue = store.state.value; if (multiple) { const currentValue = Array.isArray(selectedValue) ? selectedValue : []; const nextValue = selected ? removeItem(currentValue, value, isItemEqualToValue) : [...currentValue, value]; setValue(nextValue, createChangeEventDetails(REASONS.itemPress, event)); } else { setValue(value, createChangeEventDetails(REASONS.itemPress, event)); setOpen(false, createChangeEventDetails(REASONS.itemPress, event)); } } const defaultProps = { role: 'option', 'aria-selected': selected, 'aria-disabled': disabled || undefined, tabIndex: highlighted ? 0 : -1, onFocus() { store.set('activeIndex', index); }, onMouseEnter() { if (!keyboardActiveRef.current && store.state.selectedIndex === null) { store.set('activeIndex', index); } }, onMouseMove() { store.set('activeIndex', index); }, onMouseLeave(event) { if (keyboardActiveRef.current || isMouseWithinBounds(event)) { return; } highlightTimeout.start(0, () => { if (store.state.activeIndex === index) { store.set('activeIndex', null); } }); }, onTouchStart() { selectionRef.current = { allowSelectedMouseUp: false, allowUnselectedMouseUp: false }; }, onKeyDown(event) { lastKeyRef.current = event.key; store.set('activeIndex', index); }, onClick(event) { didPointerDownRef.current = false; // Prevent double commit on {Enter} if (event.type === 'keydown' && lastKeyRef.current === null) { return; } if (disabled || lastKeyRef.current === ' ' && typingRef.current || pointerTypeRef.current !== 'touch' && !highlighted) { return; } lastKeyRef.current = null; commitSelection(event.nativeEvent); }, onPointerEnter(event) { pointerTypeRef.current = event.pointerType; }, onPointerDown(event) { pointerTypeRef.current = event.pointerType; didPointerDownRef.current = true; }, onMouseUp(event) { if (disabled) { return; } if (didPointerDownRef.current) { didPointerDownRef.current = false; return; } const disallowSelectedMouseUp = !selectionRef.current.allowSelectedMouseUp && selected; const disallowUnselectedMouseUp = !selectionRef.current.allowUnselectedMouseUp && !selected; if (disallowSelectedMouseUp || disallowUnselectedMouseUp || pointerTypeRef.current !== 'touch' && !highlighted) { return; } commitSelection(event.nativeEvent); } }; const element = useRenderElement('div', componentProps, { ref: [buttonRef, forwardedRef, listItem.ref, itemRef], state, props: [rootProps, defaultProps, elementProps, getButtonProps] }); const contextValue = React.useMemo(() => ({ selected, indexRef, textRef, selectedByFocus, hasRegistered }), [selected, indexRef, textRef, selectedByFocus, hasRegistered]); return /*#__PURE__*/_jsx(SelectItemContext.Provider, { value: contextValue, children: element }); })); if (process.env.NODE_ENV !== "production") SelectItem.displayName = "SelectItem";