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.

106 lines 5.68 kB
import { isHTMLElement } from '@floating-ui/utils/dom'; export { stopEvent, isIndexOutOfListBounds, isListIndexDisabled, createGridCellMap, findNonDisabledListIndex, getGridCellIndexOfCorner, getGridCellIndices, getGridNavigatedIndex, getMaxListIndex, getMinListIndex } from "../floating-ui-react/utils.js"; export const ARROW_UP = 'ArrowUp'; export const ARROW_DOWN = 'ArrowDown'; export const ARROW_LEFT = 'ArrowLeft'; export const ARROW_RIGHT = 'ArrowRight'; export const HOME = 'Home'; export const END = 'End'; export const HORIZONTAL_KEYS = new Set([ARROW_LEFT, ARROW_RIGHT]); export const HORIZONTAL_KEYS_WITH_EXTRA_KEYS = new Set([ARROW_LEFT, ARROW_RIGHT, HOME, END]); export const VERTICAL_KEYS = new Set([ARROW_UP, ARROW_DOWN]); export const VERTICAL_KEYS_WITH_EXTRA_KEYS = new Set([ARROW_UP, ARROW_DOWN, HOME, END]); export const ARROW_KEYS = new Set([...HORIZONTAL_KEYS, ...VERTICAL_KEYS]); export const ALL_KEYS = new Set([...ARROW_KEYS, HOME, END]); export const COMPOSITE_KEYS = new Set([ARROW_UP, ARROW_DOWN, ARROW_LEFT, ARROW_RIGHT, HOME, END]); export const SHIFT = 'Shift'; export const CONTROL = 'Control'; export const ALT = 'Alt'; export const META = 'Meta'; export const MODIFIER_KEYS = new Set([SHIFT, CONTROL, ALT, META]); function isInputElement(element) { return isHTMLElement(element) && element.tagName === 'INPUT'; } export function isNativeInput(element) { if (isInputElement(element) && element.selectionStart != null) { return true; } if (isHTMLElement(element) && element.tagName === 'TEXTAREA') { return true; } return false; } export function scrollIntoViewIfNeeded(scrollContainer, element, direction, orientation) { if (!scrollContainer || !element || !element.scrollTo) { return; } let targetX = scrollContainer.scrollLeft; let targetY = scrollContainer.scrollTop; const isOverflowingX = scrollContainer.clientWidth < scrollContainer.scrollWidth; const isOverflowingY = scrollContainer.clientHeight < scrollContainer.scrollHeight; if (isOverflowingX && orientation !== 'vertical') { const elementOffsetLeft = getOffset(scrollContainer, element, 'left'); const containerStyles = getStyles(scrollContainer); const elementStyles = getStyles(element); if (direction === 'ltr') { if (elementOffsetLeft + element.offsetWidth + elementStyles.scrollMarginRight > scrollContainer.scrollLeft + scrollContainer.clientWidth - containerStyles.scrollPaddingRight) { // overflow to the right, scroll to align right edges targetX = elementOffsetLeft + element.offsetWidth + elementStyles.scrollMarginRight - scrollContainer.clientWidth + containerStyles.scrollPaddingRight; } else if (elementOffsetLeft - elementStyles.scrollMarginLeft < scrollContainer.scrollLeft + containerStyles.scrollPaddingLeft) { // overflow to the left, scroll to align left edges targetX = elementOffsetLeft - elementStyles.scrollMarginLeft - containerStyles.scrollPaddingLeft; } } if (direction === 'rtl') { if (elementOffsetLeft - elementStyles.scrollMarginRight < scrollContainer.scrollLeft + containerStyles.scrollPaddingLeft) { // overflow to the left, scroll to align left edges targetX = elementOffsetLeft - elementStyles.scrollMarginLeft - containerStyles.scrollPaddingLeft; } else if (elementOffsetLeft + element.offsetWidth + elementStyles.scrollMarginRight > scrollContainer.scrollLeft + scrollContainer.clientWidth - containerStyles.scrollPaddingRight) { // overflow to the right, scroll to align right edges targetX = elementOffsetLeft + element.offsetWidth + elementStyles.scrollMarginRight - scrollContainer.clientWidth + containerStyles.scrollPaddingRight; } } } if (isOverflowingY && orientation !== 'horizontal') { const elementOffsetTop = getOffset(scrollContainer, element, 'top'); const containerStyles = getStyles(scrollContainer); const elementStyles = getStyles(element); if (elementOffsetTop - elementStyles.scrollMarginTop < scrollContainer.scrollTop + containerStyles.scrollPaddingTop) { // overflow upwards, align top edges targetY = elementOffsetTop - elementStyles.scrollMarginTop - containerStyles.scrollPaddingTop; } else if (elementOffsetTop + element.offsetHeight + elementStyles.scrollMarginBottom > scrollContainer.scrollTop + scrollContainer.clientHeight - containerStyles.scrollPaddingBottom) { // overflow downwards, align bottom edges targetY = elementOffsetTop + element.offsetHeight + elementStyles.scrollMarginBottom - scrollContainer.clientHeight + containerStyles.scrollPaddingBottom; } } scrollContainer.scrollTo({ left: targetX, top: targetY, behavior: 'auto' }); } function getOffset(ancestor, element, side) { const propName = side === 'left' ? 'offsetLeft' : 'offsetTop'; let result = 0; while (element.offsetParent) { result += element[propName]; if (element.offsetParent === ancestor) { break; } element = element.offsetParent; } return result; } function getStyles(element) { const styles = getComputedStyle(element); return { scrollMarginTop: parseFloat(styles.scrollMarginTop) || 0, scrollMarginRight: parseFloat(styles.scrollMarginRight) || 0, scrollMarginBottom: parseFloat(styles.scrollMarginBottom) || 0, scrollMarginLeft: parseFloat(styles.scrollMarginLeft) || 0, scrollPaddingTop: parseFloat(styles.scrollPaddingTop) || 0, scrollPaddingRight: parseFloat(styles.scrollPaddingRight) || 0, scrollPaddingBottom: parseFloat(styles.scrollPaddingBottom) || 0, scrollPaddingLeft: parseFloat(styles.scrollPaddingLeft) || 0 }; }