UNPKG

@mui/x-data-grid

Version:

The Community plan edition of the MUI X Data Grid components.

208 lines (201 loc) 7.19 kB
'use client'; import _extends from "@babel/runtime/helpers/esm/extends"; import _objectWithoutPropertiesLoose from "@babel/runtime/helpers/esm/objectWithoutPropertiesLoose"; const _excluded = ["render", "className"]; import * as React from 'react'; import PropTypes from 'prop-types'; import { styled } from '@mui/system'; import composeClasses from '@mui/utils/composeClasses'; import clsx from 'clsx'; import { forwardRef } from '@mui/x-internals/forwardRef'; import { useComponentRenderer } from '@mui/x-internals/useComponentRenderer'; import { vars } from "../../constants/cssVariables.js"; import { getDataGridUtilityClass } from "../../constants/gridClasses.js"; import { ToolbarContext } from "./ToolbarContext.js"; import { useGridRootProps } from "../../hooks/utils/useGridRootProps.js"; import { sortByDocumentPosition } from "./utils.js"; import { jsx as _jsx } from "react/jsx-runtime"; const useUtilityClasses = ownerState => { const { classes } = ownerState; const slots = { root: ['toolbar'] }; return composeClasses(slots, getDataGridUtilityClass, classes); }; const ToolbarRoot = styled('div', { name: 'MuiDataGrid', slot: 'Toolbar' })({ flex: 0, display: 'flex', alignItems: 'center', justifyContent: 'end', gap: vars.spacing(0.25), padding: vars.spacing(0.75), minHeight: 52, boxSizing: 'border-box', borderBottom: `1px solid ${vars.colors.border.base}` }); /** * The top level Toolbar component that provides context to child components. * It renders a styled `<div />` element. * * Demos: * * - [Toolbar](https://mui.com/x/react-data-grid/components/toolbar/) * * API: * * - [Toolbar API](https://mui.com/x/api/data-grid/toolbar/) */ const Toolbar = forwardRef(function Toolbar(props, ref) { const { render, className } = props, other = _objectWithoutPropertiesLoose(props, _excluded); const rootProps = useGridRootProps(); const classes = useUtilityClasses(rootProps); const [focusableItemId, setFocusableItemId] = React.useState(null); const [items, setItems] = React.useState([]); const getSortedItems = React.useCallback(() => items.sort(sortByDocumentPosition), [items]); const findEnabledItem = React.useCallback((startIndex, step, wrap = true) => { let index = startIndex; const sortedItems = getSortedItems(); const itemCount = sortedItems.length; // Look for enabled items in the specified direction for (let i = 0; i < itemCount; i += 1) { index += step; // Handle wrapping around the ends if (index >= itemCount) { if (!wrap) { return -1; } index = 0; } else if (index < 0) { if (!wrap) { return -1; } index = itemCount - 1; } // Return if we found an enabled item if (!sortedItems[index].ref.current?.disabled && sortedItems[index].ref.current?.ariaDisabled !== 'true') { return index; } } // If we've checked all items and found none enabled return -1; }, [getSortedItems]); const registerItem = React.useCallback((id, itemRef) => { setItems(prevItems => [...prevItems, { id, ref: itemRef }]); }, []); const unregisterItem = React.useCallback(id => { setItems(prevItems => prevItems.filter(i => i.id !== id)); }, []); const onItemKeyDown = React.useCallback(event => { if (!focusableItemId) { return; } const sortedItems = getSortedItems(); const focusableItemIndex = sortedItems.findIndex(item => item.id === focusableItemId); let newIndex = -1; if (event.key === 'ArrowRight') { event.preventDefault(); newIndex = findEnabledItem(focusableItemIndex, 1); } else if (event.key === 'ArrowLeft') { event.preventDefault(); newIndex = findEnabledItem(focusableItemIndex, -1); } else if (event.key === 'Home') { event.preventDefault(); newIndex = findEnabledItem(-1, 1, false); } else if (event.key === 'End') { event.preventDefault(); newIndex = findEnabledItem(sortedItems.length, -1, false); } // TODO: Check why this is necessary if (newIndex >= 0 && newIndex < sortedItems.length) { const item = sortedItems[newIndex]; setFocusableItemId(item.id); item.ref.current?.focus(); } }, [getSortedItems, focusableItemId, findEnabledItem]); const onItemFocus = React.useCallback(id => { if (focusableItemId !== id) { setFocusableItemId(id); } }, [focusableItemId, setFocusableItemId]); const onItemDisabled = React.useCallback(id => { const sortedItems = getSortedItems(); const currentIndex = sortedItems.findIndex(item => item.id === id); const newIndex = findEnabledItem(currentIndex, 1); if (newIndex >= 0 && newIndex < sortedItems.length) { const item = sortedItems[newIndex]; setFocusableItemId(item.id); item.ref.current?.focus(); } }, [getSortedItems, findEnabledItem]); React.useEffect(() => { const sortedItems = getSortedItems(); if (sortedItems.length > 0) { // Set initial focusable item if (!focusableItemId) { setFocusableItemId(sortedItems[0].id); return; } const focusableItemIndex = sortedItems.findIndex(item => item.id === focusableItemId); if (!sortedItems[focusableItemIndex]) { // Last item has been removed from the items array const item = sortedItems[sortedItems.length - 1]; if (item) { setFocusableItemId(item.id); item.ref.current?.focus(); } } else if (focusableItemIndex === -1) { // Focused item has been removed from the items array const item = sortedItems[focusableItemIndex]; if (item) { setFocusableItemId(item.id); item.ref.current?.focus(); } } } // eslint-disable-next-line react-hooks/exhaustive-deps }, [getSortedItems, findEnabledItem]); const contextValue = React.useMemo(() => ({ focusableItemId, registerItem, unregisterItem, onItemKeyDown, onItemFocus, onItemDisabled }), [focusableItemId, registerItem, unregisterItem, onItemKeyDown, onItemFocus, onItemDisabled]); const element = useComponentRenderer(ToolbarRoot, render, _extends({ role: 'toolbar', 'aria-orientation': 'horizontal', 'aria-label': rootProps.label || undefined, className: clsx(classes.root, className) }, other, { ref })); return /*#__PURE__*/_jsx(ToolbarContext.Provider, { value: contextValue, children: element }); }); if (process.env.NODE_ENV !== "production") Toolbar.displayName = "Toolbar"; process.env.NODE_ENV !== "production" ? Toolbar.propTypes = { // ----------------------------- Warning -------------------------------- // | These PropTypes are generated from the TypeScript type definitions | // | To update them edit the TypeScript types and run "pnpm proptypes" | // ---------------------------------------------------------------------- /** * A function to customize rendering of the component. */ render: PropTypes.oneOfType([PropTypes.element, PropTypes.func]) } : void 0; export { Toolbar };