UNPKG

@base-ui/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.

193 lines (189 loc) 6.71 kB
'use client'; import * as React from 'react'; import { useIsoLayoutEffect } from '@base-ui/utils/useIsoLayoutEffect'; import { useValueAsRef } from '@base-ui/utils/useValueAsRef'; import { useStore } from '@base-ui/utils/store'; import { useSelectRootContext } from "../root/SelectRootContext.js"; import { useCompositeListItem, IndexGuessBehavior } from "../../internals/composite/list/useCompositeListItem.js"; import { useRenderElement } from "../../internals/useRenderElement.js"; import { SelectItemContext } from "./SelectItemContext.js"; import { selectors } from "../store.js"; import { useButton } from "../../internals/use-button/index.js"; import { createChangeEventDetails } from "../../internals/createBaseUIEventDetails.js"; import { REASONS } from "../../internals/reasons.js"; import { compareItemEquality, removeItem } from "../../internals/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: itemValue = null, label, disabled = false, nativeButton = false, style, ...elementProps } = componentProps; const textRef = React.useRef(null); const listItem = useCompositeListItem({ label, textRef, indexGuessBehavior: IndexGuessBehavior.GuessFromOrder }); const { store, getItemProps, setOpen, setValue, selectionRef, typingRef, valuesRef, multiple, selectedItemTextRef } = useSelectRootContext(); const highlighted = useStore(store, selectors.isActive, listItem.index); const selected = useStore(store, selectors.isSelected, listItem.index, itemValue); 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] = itemValue; return () => { delete values[index]; }; }, [hasRegistered, index, itemValue, valuesRef]); useIsoLayoutEffect(() => { if (!hasRegistered) { return; } const selectedValue = store.state.value; let selectedCandidate = selectedValue; if (multiple && Array.isArray(selectedValue) && selectedValue.length > 0) { selectedCandidate = selectedValue[selectedValue.length - 1]; } if (selectedCandidate !== undefined && compareItemEquality(itemValue, selectedCandidate, isItemEqualToValue)) { store.set('selectedIndex', index); // Make sure SelectPopup can measure the selected item on first open. // SelectItemText can still update this ref later when focus moves. if (textRef.current) { selectedItemTextRef.current = textRef.current; } } }, [hasRegistered, index, multiple, isItemEqualToValue, store, itemValue, selectedItemTextRef]); const state = { disabled, selected, highlighted }; const rootProps = getItemProps({ active: highlighted, selected }); 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, composite: true }); function commitSelection(event) { const selectedValue = store.state.value; if (multiple) { const currentValue = Array.isArray(selectedValue) ? selectedValue : []; const nextValue = selected ? removeItem(currentValue, itemValue, isItemEqualToValue) : [...currentValue, itemValue]; setValue(nextValue, createChangeEventDetails(REASONS.itemPress, event)); } else { setValue(itemValue, createChangeEventDetails(REASONS.itemPress, event)); setOpen(false, createChangeEventDetails(REASONS.itemPress, event)); } } const defaultProps = { role: 'option', 'aria-selected': selected, tabIndex: highlighted ? 0 : -1, onTouchStart() { selectionRef.current = { allowSelectedMouseUp: false, allowUnselectedMouseUp: false }; }, onKeyDown(event) { lastKeyRef.current = event.key; store.set('activeIndex', index); if (event.key === ' ' && typingRef.current) { event.preventDefault(); } }, onClick(event) { didPointerDownRef.current = false; // Prevent double commit on {Enter} if (event.type === 'keydown' && lastKeyRef.current === null) { return; } if (disabled || event.type === 'keydown' && 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() { if (disabled) { return; } // Regular click (pointerdown on this element) if didPointerDownRef is set, otherwise drag-to-select 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; } itemRef.current?.click(); } }; 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";