UNPKG

@mui/x-data-grid

Version:

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

350 lines (346 loc) 13.8 kB
import _extends from "@babel/runtime/helpers/esm/extends"; import _objectWithoutPropertiesLoose from "@babel/runtime/helpers/esm/objectWithoutPropertiesLoose"; const _excluded = ["column", "rowId", "editCellState", "align", "children", "colIndex", "width", "className", "style", "gridHasScrollX", "colSpan", "disableDragEvents", "isNotVisible", "pinnedOffset", "pinnedPosition", "sectionIndex", "sectionLength", "gridHasFiller", "onClick", "onDoubleClick", "onMouseDown", "onMouseUp", "onMouseOver", "onKeyDown", "onKeyUp", "onDragEnter", "onDragOver"], _excluded2 = ["changeReason", "unstable_updateValueOnRender"]; import * as React from 'react'; import PropTypes from 'prop-types'; import clsx from 'clsx'; import { unstable_useForkRef as useForkRef, unstable_composeClasses as composeClasses, unstable_ownerDocument as ownerDocument, unstable_capitalize as capitalize } from '@mui/utils'; import { fastMemo } from '../../utils/fastMemo'; import { doesSupportPreventScroll } from '../../utils/doesSupportPreventScroll'; import { getDataGridUtilityClass, gridClasses } from '../../constants/gridClasses'; import { GridCellModes } from '../../models'; import { useGridSelector, objectShallowCompare } from '../../hooks/utils/useGridSelector'; import { useGridApiContext } from '../../hooks/utils/useGridApiContext'; import { useGridRootProps } from '../../hooks/utils/useGridRootProps'; import { gridFocusCellSelector } from '../../hooks/features/focus/gridFocusStateSelector'; import { MissingRowIdError } from '../../hooks/features/rows/useGridParamsApi'; import { shouldCellShowLeftBorder, shouldCellShowRightBorder } from '../../utils/cellBorderUtils'; import { GridPinnedColumnPosition } from '../../hooks/features/columns/gridColumnsInterfaces'; import { jsx as _jsx } from "react/jsx-runtime"; export let PinnedPosition = /*#__PURE__*/function (PinnedPosition) { PinnedPosition[PinnedPosition["NONE"] = 0] = "NONE"; PinnedPosition[PinnedPosition["LEFT"] = 1] = "LEFT"; PinnedPosition[PinnedPosition["RIGHT"] = 2] = "RIGHT"; PinnedPosition[PinnedPosition["VIRTUAL"] = 3] = "VIRTUAL"; return PinnedPosition; }({}); export const gridPinnedColumnPositionLookup = { [PinnedPosition.LEFT]: GridPinnedColumnPosition.LEFT, [PinnedPosition.RIGHT]: GridPinnedColumnPosition.RIGHT, [PinnedPosition.NONE]: undefined, [PinnedPosition.VIRTUAL]: undefined }; const EMPTY_CELL_PARAMS = { id: -1, field: '__unset__', row: {}, rowNode: { id: -1, depth: 0, type: 'leaf', parent: -1, groupingKey: null }, colDef: { type: 'string', field: '__unset__', computedWidth: 0 }, cellMode: GridCellModes.View, hasFocus: false, tabIndex: -1, value: null, formattedValue: '__unset__', isEditable: false, api: {} }; const useUtilityClasses = ownerState => { const { align, showLeftBorder, showRightBorder, pinnedPosition, isEditable, isSelected, isSelectionMode, classes } = ownerState; const slots = { root: ['cell', `cell--text${capitalize(align)}`, isSelected && 'selected', isEditable && 'cell--editable', showLeftBorder && 'cell--withLeftBorder', showRightBorder && 'cell--withRightBorder', pinnedPosition === PinnedPosition.LEFT && 'cell--pinnedLeft', pinnedPosition === PinnedPosition.RIGHT && 'cell--pinnedRight', isSelectionMode && !isEditable && 'cell--selectionMode'] }; return composeClasses(slots, getDataGridUtilityClass, classes); }; let warnedOnce = false; // TODO(v7): Removing the wrapper will break the docs performance visualization demo. const GridCell = /*#__PURE__*/React.forwardRef((props, ref) => { const { column, rowId, editCellState, align, colIndex, width, className, style: styleProp, colSpan, disableDragEvents, isNotVisible, pinnedOffset, pinnedPosition, sectionIndex, sectionLength, gridHasFiller, onClick, onDoubleClick, onMouseDown, onMouseUp, onMouseOver, onKeyDown, onKeyUp, onDragEnter, onDragOver } = props, other = _objectWithoutPropertiesLoose(props, _excluded); const apiRef = useGridApiContext(); const rootProps = useGridRootProps(); const field = column.field; const cellParamsWithAPI = useGridSelector(apiRef, () => { // This is required because `.getCellParams` tries to get the `state.rows.tree` entry // associated with `rowId`/`fieldId`, but this selector runs after the state has been // updated, while `rowId`/`fieldId` reference an entry in the old state. try { const cellParams = apiRef.current.getCellParams(rowId, field); const result = cellParams; result.api = apiRef.current; return result; } catch (e) { if (e instanceof MissingRowIdError) { return EMPTY_CELL_PARAMS; } throw e; } }, objectShallowCompare); const isSelected = useGridSelector(apiRef, () => apiRef.current.unstable_applyPipeProcessors('isCellSelected', false, { id: rowId, field })); const { cellMode, hasFocus, isEditable = false, value } = cellParamsWithAPI; const canManageOwnFocus = column.type === 'actions' && column.getActions?.(apiRef.current.getRowParams(rowId)).some(action => !action.props.disabled); const tabIndex = (cellMode === 'view' || !isEditable) && !canManageOwnFocus ? cellParamsWithAPI.tabIndex : -1; const { classes: rootClasses, getCellClassName } = rootProps; // There is a hidden grid state access in `applyPipeProcessor('cellClassName', ...)` const pipesClassName = useGridSelector(apiRef, () => apiRef.current.unstable_applyPipeProcessors('cellClassName', [], { id: rowId, field }).filter(Boolean).join(' ')); const classNames = [pipesClassName]; if (column.cellClassName) { classNames.push(typeof column.cellClassName === 'function' ? column.cellClassName(cellParamsWithAPI) : column.cellClassName); } if (column.display === 'flex') { classNames.push(gridClasses['cell--flex']); } if (getCellClassName) { classNames.push(getCellClassName(cellParamsWithAPI)); } const valueToRender = cellParamsWithAPI.formattedValue ?? value; const cellRef = React.useRef(null); const handleRef = useForkRef(ref, cellRef); const focusElementRef = React.useRef(null); const isSelectionMode = rootProps.cellSelection ?? false; const position = gridPinnedColumnPositionLookup[pinnedPosition]; const showLeftBorder = shouldCellShowLeftBorder(position, sectionIndex); const showRightBorder = shouldCellShowRightBorder(position, sectionIndex, sectionLength, rootProps.showCellVerticalBorder, gridHasFiller); const ownerState = { align, showLeftBorder, showRightBorder, isEditable, classes: rootProps.classes, pinnedPosition, isSelected, isSelectionMode }; const classes = useUtilityClasses(ownerState); const publishMouseUp = React.useCallback(eventName => event => { const params = apiRef.current.getCellParams(rowId, field || ''); apiRef.current.publishEvent(eventName, params, event); if (onMouseUp) { onMouseUp(event); } }, [apiRef, field, onMouseUp, rowId]); const publishMouseDown = React.useCallback(eventName => event => { const params = apiRef.current.getCellParams(rowId, field || ''); apiRef.current.publishEvent(eventName, params, event); if (onMouseDown) { onMouseDown(event); } }, [apiRef, field, onMouseDown, rowId]); const publish = React.useCallback((eventName, propHandler) => event => { // The row might have been deleted during the click if (!apiRef.current.getRow(rowId)) { return; } const params = apiRef.current.getCellParams(rowId, field || ''); apiRef.current.publishEvent(eventName, params, event); if (propHandler) { propHandler(event); } }, [apiRef, field, rowId]); const style = React.useMemo(() => { if (isNotVisible) { return { padding: 0, opacity: 0, width: 0, border: 0 }; } const cellStyle = _extends({ '--width': `${width}px` }, styleProp); if (pinnedPosition === PinnedPosition.LEFT) { cellStyle.left = pinnedOffset; } if (pinnedPosition === PinnedPosition.RIGHT) { cellStyle.right = pinnedOffset; } return cellStyle; }, [width, isNotVisible, styleProp, pinnedOffset, pinnedPosition]); React.useEffect(() => { if (!hasFocus || cellMode === GridCellModes.Edit) { return; } const doc = ownerDocument(apiRef.current.rootElementRef.current); if (cellRef.current && !cellRef.current.contains(doc.activeElement)) { const focusableElement = cellRef.current.querySelector('[tabindex="0"]'); const elementToFocus = focusElementRef.current || focusableElement || cellRef.current; if (doesSupportPreventScroll()) { elementToFocus.focus({ preventScroll: true }); } else { const scrollPosition = apiRef.current.getScrollPosition(); elementToFocus.focus(); apiRef.current.scroll(scrollPosition); } } }, [hasFocus, cellMode, apiRef]); if (cellParamsWithAPI === EMPTY_CELL_PARAMS) { return null; } let handleFocus = other.onFocus; if (process.env.NODE_ENV === 'test' && rootProps.experimentalFeatures?.warnIfFocusStateIsNotSynced) { handleFocus = event => { const focusedCell = gridFocusCellSelector(apiRef); if (focusedCell?.id === rowId && focusedCell.field === field) { if (typeof other.onFocus === 'function') { other.onFocus(event); } return; } if (!warnedOnce) { console.warn([`MUI X: The cell with id=${rowId} and field=${field} received focus.`, `According to the state, the focus should be at id=${focusedCell?.id}, field=${focusedCell?.field}.`, "Not syncing the state may cause unwanted behaviors since the `cellFocusIn` event won't be fired.", 'Call `fireEvent.mouseUp` before the `fireEvent.click` to sync the focus with the state.'].join('\n')); warnedOnce = true; } }; } let children; let title; if (editCellState === null && column.renderCell) { children = column.renderCell(cellParamsWithAPI); } if (editCellState !== null && column.renderEditCell) { const updatedRow = apiRef.current.getRowWithUpdatedValues(rowId, column.field); // eslint-disable-next-line @typescript-eslint/naming-convention const editCellStateRest = _objectWithoutPropertiesLoose(editCellState, _excluded2); const formattedValue = column.valueFormatter ? column.valueFormatter(editCellState.value, updatedRow, column, apiRef) : cellParamsWithAPI.formattedValue; const params = _extends({}, cellParamsWithAPI, { row: updatedRow, formattedValue }, editCellStateRest); children = column.renderEditCell(params); classNames.push(gridClasses['cell--editing']); classNames.push(rootClasses?.['cell--editing']); } if (children === undefined) { const valueString = valueToRender?.toString(); children = valueString; title = valueString; } if ( /*#__PURE__*/React.isValidElement(children) && canManageOwnFocus) { children = /*#__PURE__*/React.cloneElement(children, { focusElementRef }); } const draggableEventHandlers = disableDragEvents ? null : { onDragEnter: publish('cellDragEnter', onDragEnter), onDragOver: publish('cellDragOver', onDragOver) }; return /*#__PURE__*/_jsx("div", _extends({ ref: handleRef, className: clsx(className, classNames, classes.root), role: "gridcell", "data-field": field, "data-colindex": colIndex, "aria-colindex": colIndex + 1, "aria-colspan": colSpan, style: style, title: title, tabIndex: tabIndex, onClick: publish('cellClick', onClick), onDoubleClick: publish('cellDoubleClick', onDoubleClick), onMouseOver: publish('cellMouseOver', onMouseOver), onMouseDown: publishMouseDown('cellMouseDown'), onMouseUp: publishMouseUp('cellMouseUp'), onKeyDown: publish('cellKeyDown', onKeyDown), onKeyUp: publish('cellKeyUp', onKeyUp) }, draggableEventHandlers, other, { onFocus: handleFocus, children: children })); }); process.env.NODE_ENV !== "production" ? GridCell.propTypes = { // ----------------------------- Warning -------------------------------- // | These PropTypes are generated from the TypeScript type definitions | // | To update them edit the TypeScript types and run "yarn proptypes" | // ---------------------------------------------------------------------- align: PropTypes.oneOf(['center', 'left', 'right']).isRequired, className: PropTypes.string, colIndex: PropTypes.number.isRequired, colSpan: PropTypes.number, column: PropTypes.object.isRequired, disableDragEvents: PropTypes.bool, editCellState: PropTypes.shape({ changeReason: PropTypes.oneOf(['debouncedSetEditCellValue', 'setEditCellValue']), isProcessingProps: PropTypes.bool, isValidating: PropTypes.bool, value: PropTypes.any }), gridHasFiller: PropTypes.bool.isRequired, isNotVisible: PropTypes.bool.isRequired, onClick: PropTypes.func, onDoubleClick: PropTypes.func, onDragEnter: PropTypes.func, onDragOver: PropTypes.func, onKeyDown: PropTypes.func, onMouseDown: PropTypes.func, onMouseUp: PropTypes.func, pinnedOffset: PropTypes.number.isRequired, pinnedPosition: PropTypes.oneOf([0, 1, 2, 3]).isRequired, rowId: PropTypes.oneOfType([PropTypes.number, PropTypes.string]).isRequired, sectionIndex: PropTypes.number.isRequired, sectionLength: PropTypes.number.isRequired, width: PropTypes.number.isRequired } : void 0; const MemoizedGridCell = fastMemo(GridCell); export { MemoizedGridCell as GridCell };