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.

384 lines (379 loc) 12.6 kB
import { floor } from '@floating-ui/utils'; import { getComputedStyle } from '@floating-ui/utils/dom'; import { stopEvent } from "./event.js"; import { ARROW_DOWN, ARROW_LEFT, ARROW_RIGHT, ARROW_UP } from "./constants.js"; export function isDifferentGridRow(index, cols, prevRow) { return Math.floor(index / cols) !== prevRow; } export function isIndexOutOfListBounds(list, index) { return index < 0 || index >= list.length; } export function getMinListIndex(listRef, disabledIndices) { return findNonDisabledListIndex(listRef.current, { disabledIndices }); } export function getMaxListIndex(listRef, disabledIndices) { return findNonDisabledListIndex(listRef.current, { decrement: true, startingIndex: listRef.current.length, disabledIndices }); } export function findNonDisabledListIndex(list, { startingIndex = -1, decrement = false, disabledIndices, amount = 1 } = {}) { let index = startingIndex; do { index += decrement ? -amount : amount; } while (index >= 0 && index <= list.length - 1 && isListIndexDisabled(list, index, disabledIndices)); return index; } export function getGridNavigatedIndex(list, { event, orientation, loopFocus, onLoop, rtl, cols, disabledIndices, minIndex, maxIndex, prevIndex, stopEvent: stop = false }) { let nextIndex = prevIndex; let verticalDirection; if (event.key === ARROW_UP) { verticalDirection = 'up'; } else if (event.key === ARROW_DOWN) { verticalDirection = 'down'; } if (verticalDirection) { // ------------------------------------------------------------------------- // Detect row structure only when handling vertical navigation. This keeps // the non-vertical key paths free from row inference work. // ------------------------------------------------------------------------- const rows = []; const rowIndexMap = []; let hasRoleRow = false; let visibleItemCount = 0; { let currentRowEl = null; let currentRowIndex = -1; list.forEach((el, idx) => { if (el == null) { return; } visibleItemCount += 1; const rowEl = el.closest('[role="row"]'); if (rowEl) { hasRoleRow = true; } if (rowEl !== currentRowEl || currentRowIndex === -1) { currentRowEl = rowEl; currentRowIndex += 1; rows[currentRowIndex] = []; } rows[currentRowIndex].push(idx); rowIndexMap[idx] = currentRowIndex; }); } let hasDomRows = false; let inferredDomCols = 0; if (hasRoleRow) { for (const row of rows) { const rowLength = row.length; if (rowLength > inferredDomCols) { inferredDomCols = rowLength; } if (rowLength !== cols) { hasDomRows = true; } } } const hasVirtualizedGaps = hasDomRows && visibleItemCount < list.length; const verticalCols = inferredDomCols || cols; const navigateVertically = direction => { if (!hasDomRows || prevIndex === -1) { return undefined; } const currentRow = rowIndexMap[prevIndex]; if (currentRow == null) { return undefined; } const colInRow = rows[currentRow].indexOf(prevIndex); const step = direction === 'up' ? -1 : 1; for (let nextRow = currentRow + step, i = 0; i < rows.length; i += 1, nextRow += step) { if (nextRow < 0 || nextRow >= rows.length) { if (!loopFocus || hasVirtualizedGaps) { return undefined; } nextRow = nextRow < 0 ? rows.length - 1 : 0; if (onLoop) { const clampedCol = Math.min(colInRow, rows[nextRow].length - 1); const targetItemIndex = rows[nextRow][clampedCol] ?? rows[nextRow][0]; const returnedItemIndex = onLoop(event, prevIndex, targetItemIndex); nextRow = rowIndexMap[returnedItemIndex] ?? nextRow; } } const targetRow = rows[nextRow]; for (let col = Math.min(colInRow, targetRow.length - 1); col >= 0; col -= 1) { const candidate = targetRow[col]; if (!isListIndexDisabled(list, candidate, disabledIndices)) { return candidate; } } } return undefined; }; const navigateVerticallyWithInferredRows = direction => { if (!hasVirtualizedGaps || prevIndex === -1) { return undefined; } const colInRow = prevIndex % verticalCols; const rowStep = direction === 'up' ? -verticalCols : verticalCols; const lastRowStart = maxIndex - maxIndex % verticalCols; const rowCount = floor(maxIndex / verticalCols) + 1; for (let rowStart = prevIndex - colInRow + rowStep, i = 0; i < rowCount; i += 1, rowStart += rowStep) { if (rowStart < 0 || rowStart > maxIndex) { if (!loopFocus) { return undefined; } rowStart = rowStart < 0 ? lastRowStart : 0; } const rowEnd = Math.min(rowStart + verticalCols - 1, maxIndex); for (let candidate = Math.min(rowStart + colInRow, rowEnd); candidate >= rowStart; candidate -= 1) { if (!isListIndexDisabled(list, candidate, disabledIndices)) { return candidate; } } } return undefined; }; if (stop) { stopEvent(event); } const verticalCandidate = navigateVertically(verticalDirection) ?? navigateVerticallyWithInferredRows(verticalDirection); if (verticalCandidate !== undefined) { nextIndex = verticalCandidate; } else if (prevIndex === -1) { nextIndex = verticalDirection === 'up' ? maxIndex : minIndex; } else { nextIndex = findNonDisabledListIndex(list, { startingIndex: prevIndex, amount: verticalCols, decrement: verticalDirection === 'up', disabledIndices }); if (loopFocus) { if (verticalDirection === 'up' && (prevIndex - verticalCols < minIndex || nextIndex < 0)) { const col = prevIndex % verticalCols; const maxCol = maxIndex % verticalCols; const offset = maxIndex - (maxCol - col); if (maxCol === col) { nextIndex = maxIndex; } else { nextIndex = maxCol > col ? offset : offset - verticalCols; } if (onLoop) { nextIndex = onLoop(event, prevIndex, nextIndex); } } if (verticalDirection === 'down' && prevIndex + verticalCols > maxIndex) { nextIndex = findNonDisabledListIndex(list, { startingIndex: prevIndex % verticalCols - verticalCols, amount: verticalCols, disabledIndices }); if (onLoop) { nextIndex = onLoop(event, prevIndex, nextIndex); } } } } if (isIndexOutOfListBounds(list, nextIndex)) { nextIndex = prevIndex; } } // Remains on the same row/column. if (orientation === 'both') { const prevRow = floor(prevIndex / cols); if (event.key === (rtl ? ARROW_LEFT : ARROW_RIGHT)) { if (stop) { stopEvent(event); } if (prevIndex % cols !== cols - 1) { nextIndex = findNonDisabledListIndex(list, { startingIndex: prevIndex, disabledIndices }); if (loopFocus && isDifferentGridRow(nextIndex, cols, prevRow)) { nextIndex = findNonDisabledListIndex(list, { startingIndex: prevIndex - prevIndex % cols - 1, disabledIndices }); if (onLoop) { nextIndex = onLoop(event, prevIndex, nextIndex); } } } else if (loopFocus) { nextIndex = findNonDisabledListIndex(list, { startingIndex: prevIndex - prevIndex % cols - 1, disabledIndices }); if (onLoop) { nextIndex = onLoop(event, prevIndex, nextIndex); } } if (isDifferentGridRow(nextIndex, cols, prevRow)) { nextIndex = prevIndex; } } if (event.key === (rtl ? ARROW_RIGHT : ARROW_LEFT)) { if (stop) { stopEvent(event); } if (prevIndex % cols !== 0) { nextIndex = findNonDisabledListIndex(list, { startingIndex: prevIndex, decrement: true, disabledIndices }); if (loopFocus && isDifferentGridRow(nextIndex, cols, prevRow)) { nextIndex = findNonDisabledListIndex(list, { startingIndex: prevIndex + (cols - prevIndex % cols), decrement: true, disabledIndices }); if (onLoop) { nextIndex = onLoop(event, prevIndex, nextIndex); } } } else if (loopFocus) { nextIndex = findNonDisabledListIndex(list, { startingIndex: prevIndex + (cols - prevIndex % cols), decrement: true, disabledIndices }); if (onLoop) { nextIndex = onLoop(event, prevIndex, nextIndex); } } if (isDifferentGridRow(nextIndex, cols, prevRow)) { nextIndex = prevIndex; } } const lastRow = floor(maxIndex / cols) === prevRow; if (isIndexOutOfListBounds(list, nextIndex)) { if (loopFocus && lastRow) { nextIndex = event.key === (rtl ? ARROW_RIGHT : ARROW_LEFT) ? maxIndex : findNonDisabledListIndex(list, { startingIndex: prevIndex - prevIndex % cols - 1, disabledIndices }); if (onLoop) { nextIndex = onLoop(event, prevIndex, nextIndex); } } else { nextIndex = prevIndex; } } } return nextIndex; } /** For each cell index, gets the item index that occupies that cell */ export function createGridCellMap(sizes, cols, dense) { const cellMap = []; let startIndex = 0; sizes.forEach(({ width, height }, index) => { if (width > cols) { if (process.env.NODE_ENV !== 'production') { throw new Error(`[Floating UI]: Invalid grid - item width at index ${index} is greater than grid columns`); } } let itemPlaced = false; if (dense) { startIndex = 0; } while (!itemPlaced) { const targetCells = []; for (let i = 0; i < width; i += 1) { for (let j = 0; j < height; j += 1) { targetCells.push(startIndex + i + j * cols); } } if (startIndex % cols + width <= cols && targetCells.every(cell => cellMap[cell] == null)) { targetCells.forEach(cell => { cellMap[cell] = index; }); itemPlaced = true; } else { startIndex += 1; } } }); // convert into a non-sparse array return [...cellMap]; } /** Gets cell index of an item's corner or -1 when index is -1. */ export function getGridCellIndexOfCorner(index, sizes, cellMap, cols, corner) { if (index === -1) { return -1; } const firstCellIndex = cellMap.indexOf(index); const sizeItem = sizes[index]; switch (corner) { case 'tl': return firstCellIndex; case 'tr': if (!sizeItem) { return firstCellIndex; } return firstCellIndex + sizeItem.width - 1; case 'bl': if (!sizeItem) { return firstCellIndex; } return firstCellIndex + (sizeItem.height - 1) * cols; case 'br': return cellMap.lastIndexOf(index); default: return -1; } } /** Gets all cell indices that correspond to the specified indices */ export function getGridCellIndices(indices, cellMap) { return cellMap.flatMap((index, cellIndex) => indices.includes(index) ? [cellIndex] : []); } export function isListIndexDisabled(list, index, disabledIndices) { const isExplicitlyDisabled = typeof disabledIndices === 'function' ? disabledIndices(index) : disabledIndices?.includes(index) ?? false; if (isExplicitlyDisabled) { return true; } const element = list[index]; if (!element) { return false; } if (!isElementVisible(element)) { return true; } return !disabledIndices && (element.hasAttribute('disabled') || element.getAttribute('aria-disabled') === 'true'); } export function isHiddenByStyles(styles) { return styles.visibility === 'hidden' || styles.visibility === 'collapse'; } export function isElementVisible(element, styles = element ? getComputedStyle(element) : null) { if (!element || !element.isConnected || !styles || isHiddenByStyles(styles)) { return false; } if (typeof element.checkVisibility === 'function') { return element.checkVisibility(); } return styles.display !== 'none' && styles.display !== 'contents'; }