From d8d7cf98af58e3369f9accb72223bcc808f3ebc7 Mon Sep 17 00:00:00 2001 From: Matheus Wichman Date: Tue, 30 Nov 2021 13:25:12 -0300 Subject: [PATCH] [DataGrid] Add support for `sx` prop (#3281) --- .../api-docs/data-grid/data-grid-pro.json | 6 ++++ docs/pages/api-docs/data-grid/data-grid.json | 6 ++++ .../api/buildComponentDocumentation.ts | 2 ++ .../components/data-grid/style/SxProp.js | 27 ++++++++++++++ .../components/data-grid/style/SxProp.tsx | 27 ++++++++++++++ .../data-grid/style/SxProp.tsx.preview | 11 ++++++ .../pages/components/data-grid/style/style.md | 13 +++++++ .../api-docs/data-grid/data-grid-pro-pt.json | 1 + .../api-docs/data-grid/data-grid-pro-zh.json | 1 + .../api-docs/data-grid/data-grid-pro.json | 1 + .../api-docs/data-grid/data-grid-pt.json | 3 +- .../api-docs/data-grid/data-grid-zh.json | 3 +- .../api-docs/data-grid/data-grid.json | 3 +- .../grid/_modules_/grid/GridComponentProps.ts | 18 +++++----- .../grid/components/containers/GridRoot.tsx | 35 +++++++++++++++---- packages/grid/x-data-grid-pro/package.json | 1 + .../grid/x-data-grid-pro/src/DataGridPro.tsx | 14 ++++---- .../src/tests/DataGridPro.spec.tsx | 6 ++++ .../src/tests/layout.DataGridPro.test.tsx | 25 ++++++++++++- packages/grid/x-data-grid/package.json | 1 + packages/grid/x-data-grid/src/DataGrid.tsx | 14 ++++---- .../x-data-grid/src/tests/DataGrid.spec.tsx | 4 +++ .../src/tests/layout.DataGrid.test.tsx | 23 ++++++++++++ scripts/exportsSnapshot.json | 2 +- 24 files changed, 211 insertions(+), 36 deletions(-) create mode 100644 docs/src/pages/components/data-grid/style/SxProp.js create mode 100644 docs/src/pages/components/data-grid/style/SxProp.tsx create mode 100644 docs/src/pages/components/data-grid/style/SxProp.tsx.preview create mode 100644 packages/grid/x-data-grid-pro/src/tests/DataGridPro.spec.tsx diff --git a/docs/pages/api-docs/data-grid/data-grid-pro.json b/docs/pages/api-docs/data-grid/data-grid-pro.json index 6b8fcd60a101d..9400e56697682 100644 --- a/docs/pages/api-docs/data-grid/data-grid-pro.json +++ b/docs/pages/api-docs/data-grid/data-grid-pro.json @@ -171,6 +171,12 @@ "description": "Array<{ field: string, sort?: 'asc'
| 'desc' }>" } }, + "sx": { + "type": { + "name": "union", + "description": "Array<func
| object>
| func
| object" + } + }, "throttleRowsMs": { "type": { "name": "number" }, "default": "0" }, "treeData": { "type": { "name": "bool" } } }, diff --git a/docs/pages/api-docs/data-grid/data-grid.json b/docs/pages/api-docs/data-grid/data-grid.json index 20359fee8e2e3..92e63124b10ca 100644 --- a/docs/pages/api-docs/data-grid/data-grid.json +++ b/docs/pages/api-docs/data-grid/data-grid.json @@ -140,6 +140,12 @@ "name": "arrayOf", "description": "Array<{ field: string, sort?: 'asc'
| 'desc' }>" } + }, + "sx": { + "type": { + "name": "union", + "description": "Array<func
| object>
| func
| object" + } } }, "slots": { diff --git a/docs/scripts/api/buildComponentDocumentation.ts b/docs/scripts/api/buildComponentDocumentation.ts index 67b63d555c319..5cf457d00daf3 100644 --- a/docs/scripts/api/buildComponentDocumentation.ts +++ b/docs/scripts/api/buildComponentDocumentation.ts @@ -264,6 +264,8 @@ export default async function buildComponentDocumentation( if (propName === 'classes') { description += ' See CSS API below for more details.'; + } else if (propName === 'sx') { + description += ' See the `sx` page for more details.'; } componentApi.propDescriptions[propName] = linkify(description, documentedInterfaces, 'html'); diff --git a/docs/src/pages/components/data-grid/style/SxProp.js b/docs/src/pages/components/data-grid/style/SxProp.js new file mode 100644 index 0000000000000..4d2afca44ce2d --- /dev/null +++ b/docs/src/pages/components/data-grid/style/SxProp.js @@ -0,0 +1,27 @@ +import * as React from 'react'; +import { DataGrid } from '@mui/x-data-grid'; +import { useDemoData } from '@mui/x-data-grid-generator'; + +export default function SxProp() { + const { data } = useDemoData({ + dataSet: 'Commodity', + rowLength: 20, + maxColumns: 5, + }); + + return ( +
+ +
+ ); +} diff --git a/docs/src/pages/components/data-grid/style/SxProp.tsx b/docs/src/pages/components/data-grid/style/SxProp.tsx new file mode 100644 index 0000000000000..4d2afca44ce2d --- /dev/null +++ b/docs/src/pages/components/data-grid/style/SxProp.tsx @@ -0,0 +1,27 @@ +import * as React from 'react'; +import { DataGrid } from '@mui/x-data-grid'; +import { useDemoData } from '@mui/x-data-grid-generator'; + +export default function SxProp() { + const { data } = useDemoData({ + dataSet: 'Commodity', + rowLength: 20, + maxColumns: 5, + }); + + return ( +
+ +
+ ); +} diff --git a/docs/src/pages/components/data-grid/style/SxProp.tsx.preview b/docs/src/pages/components/data-grid/style/SxProp.tsx.preview new file mode 100644 index 0000000000000..d2740fdec8ce3 --- /dev/null +++ b/docs/src/pages/components/data-grid/style/SxProp.tsx.preview @@ -0,0 +1,11 @@ + \ No newline at end of file diff --git a/docs/src/pages/components/data-grid/style/style.md b/docs/src/pages/components/data-grid/style/style.md index 735319d108095..0a6d1a48eae73 100644 --- a/docs/src/pages/components/data-grid/style/style.md +++ b/docs/src/pages/components/data-grid/style/style.md @@ -6,6 +6,19 @@ title: Data Grid - Styling

The grid CSS can be easily overwritten.

+## Using the `sx` prop + +For one-off styles, the `sx` prop can be used. +It allows to apply simple to complex customizations directly onto the `DataGrid` element. +The keys accepted can be any CSS property as well as the custom properties provided by MUI. +For more details, visit the [`sx` prop page](/system/the-sx-prop/). + +```tsx + // Sets the margin to 2 times the spacing unit = 16px +``` + +{{"demo": "pages/components/data-grid/style/SxProp.js", "bg": "inline"}} + ## Styling column headers The `GridColDef` type has properties to apply class names and custom CSS on the header. diff --git a/docs/translations/api-docs/data-grid/data-grid-pro-pt.json b/docs/translations/api-docs/data-grid/data-grid-pro-pt.json index 883d73950b4bb..e5dd929039a4f 100644 --- a/docs/translations/api-docs/data-grid/data-grid-pro-pt.json +++ b/docs/translations/api-docs/data-grid/data-grid-pro-pt.json @@ -105,6 +105,7 @@ "sortingMode": "Sorting can be processed on the server or client-side. Set it to 'client' if you would like to handle sorting on the client-side. Set it to 'server' if you would like to handle sorting on the server-side.", "sortingOrder": "The order of the sorting sequence.", "sortModel": "Set the sort model of the grid.", + "sx": "The system prop that allows defining system overrides as well as additional CSS styles. See the `sx` page for more details.", "throttleRowsMs": "If positive, the Grid will throttle updates coming from apiRef.current.updateRows and apiRef.current.setRows. It can be useful if you have a high update rate but do not want to do heavy work like filtering / sorting or rendering on each individual update.", "treeData": "If true, the rows will be gathered in a tree structure according to the getTreeDataPath prop." }, diff --git a/docs/translations/api-docs/data-grid/data-grid-pro-zh.json b/docs/translations/api-docs/data-grid/data-grid-pro-zh.json index 883d73950b4bb..e5dd929039a4f 100644 --- a/docs/translations/api-docs/data-grid/data-grid-pro-zh.json +++ b/docs/translations/api-docs/data-grid/data-grid-pro-zh.json @@ -105,6 +105,7 @@ "sortingMode": "Sorting can be processed on the server or client-side. Set it to 'client' if you would like to handle sorting on the client-side. Set it to 'server' if you would like to handle sorting on the server-side.", "sortingOrder": "The order of the sorting sequence.", "sortModel": "Set the sort model of the grid.", + "sx": "The system prop that allows defining system overrides as well as additional CSS styles. See the `sx` page for more details.", "throttleRowsMs": "If positive, the Grid will throttle updates coming from apiRef.current.updateRows and apiRef.current.setRows. It can be useful if you have a high update rate but do not want to do heavy work like filtering / sorting or rendering on each individual update.", "treeData": "If true, the rows will be gathered in a tree structure according to the getTreeDataPath prop." }, diff --git a/docs/translations/api-docs/data-grid/data-grid-pro.json b/docs/translations/api-docs/data-grid/data-grid-pro.json index 883d73950b4bb..e5dd929039a4f 100644 --- a/docs/translations/api-docs/data-grid/data-grid-pro.json +++ b/docs/translations/api-docs/data-grid/data-grid-pro.json @@ -105,6 +105,7 @@ "sortingMode": "Sorting can be processed on the server or client-side. Set it to 'client' if you would like to handle sorting on the client-side. Set it to 'server' if you would like to handle sorting on the server-side.", "sortingOrder": "The order of the sorting sequence.", "sortModel": "Set the sort model of the grid.", + "sx": "The system prop that allows defining system overrides as well as additional CSS styles. See the `sx` page for more details.", "throttleRowsMs": "If positive, the Grid will throttle updates coming from apiRef.current.updateRows and apiRef.current.setRows. It can be useful if you have a high update rate but do not want to do heavy work like filtering / sorting or rendering on each individual update.", "treeData": "If true, the rows will be gathered in a tree structure according to the getTreeDataPath prop." }, diff --git a/docs/translations/api-docs/data-grid/data-grid-pt.json b/docs/translations/api-docs/data-grid/data-grid-pt.json index dca9dbf58e6ea..cafc0128bc9f5 100644 --- a/docs/translations/api-docs/data-grid/data-grid-pt.json +++ b/docs/translations/api-docs/data-grid/data-grid-pt.json @@ -88,7 +88,8 @@ "showColumnRightBorder": "If true, the right border of the column headers are displayed.", "sortingMode": "Sorting can be processed on the server or client-side. Set it to 'client' if you would like to handle sorting on the client-side. Set it to 'server' if you would like to handle sorting on the server-side.", "sortingOrder": "The order of the sorting sequence.", - "sortModel": "Set the sort model of the grid." + "sortModel": "Set the sort model of the grid.", + "sx": "The system prop that allows defining system overrides as well as additional CSS styles. See the `sx` page for more details." }, "classDescriptions": { "autoHeight": { diff --git a/docs/translations/api-docs/data-grid/data-grid-zh.json b/docs/translations/api-docs/data-grid/data-grid-zh.json index dca9dbf58e6ea..cafc0128bc9f5 100644 --- a/docs/translations/api-docs/data-grid/data-grid-zh.json +++ b/docs/translations/api-docs/data-grid/data-grid-zh.json @@ -88,7 +88,8 @@ "showColumnRightBorder": "If true, the right border of the column headers are displayed.", "sortingMode": "Sorting can be processed on the server or client-side. Set it to 'client' if you would like to handle sorting on the client-side. Set it to 'server' if you would like to handle sorting on the server-side.", "sortingOrder": "The order of the sorting sequence.", - "sortModel": "Set the sort model of the grid." + "sortModel": "Set the sort model of the grid.", + "sx": "The system prop that allows defining system overrides as well as additional CSS styles. See the `sx` page for more details." }, "classDescriptions": { "autoHeight": { diff --git a/docs/translations/api-docs/data-grid/data-grid.json b/docs/translations/api-docs/data-grid/data-grid.json index dca9dbf58e6ea..cafc0128bc9f5 100644 --- a/docs/translations/api-docs/data-grid/data-grid.json +++ b/docs/translations/api-docs/data-grid/data-grid.json @@ -88,7 +88,8 @@ "showColumnRightBorder": "If true, the right border of the column headers are displayed.", "sortingMode": "Sorting can be processed on the server or client-side. Set it to 'client' if you would like to handle sorting on the client-side. Set it to 'server' if you would like to handle sorting on the server-side.", "sortingOrder": "The order of the sorting sequence.", - "sortModel": "Set the sort model of the grid." + "sortModel": "Set the sort model of the grid.", + "sx": "The system prop that allows defining system overrides as well as additional CSS styles. See the `sx` page for more details." }, "classDescriptions": { "autoHeight": { diff --git a/packages/grid/_modules_/grid/GridComponentProps.ts b/packages/grid/_modules_/grid/GridComponentProps.ts index 1de17b147bb5f..57dbe383c72ad 100644 --- a/packages/grid/_modules_/grid/GridComponentProps.ts +++ b/packages/grid/_modules_/grid/GridComponentProps.ts @@ -1,4 +1,6 @@ -import * as React from 'react'; +import { CommonProps } from '@mui/material/OverridableComponent'; +import { SxProps } from '@mui/system'; +import { Theme } from '@mui/material/styles'; import { GridInitialState } from './models/gridState'; import { GridApiRef } from './models/api/gridApiRef'; import { @@ -40,7 +42,7 @@ export interface GridComponentProps GridProcessedMergedOptions, GridComponentOtherProps {} -interface GridComponentOtherProps { +interface GridComponentOtherProps extends CommonProps { /** * The ref object that allows grid manipulation. Can be instantiated with [[useGridApiRef()]]. */ @@ -361,10 +363,6 @@ interface GridComponentOtherProps { * The id of the element containing a label for the grid. */ 'aria-labelledby'?: string; - /** - * @ignore - do not document - */ - className?: string; /** * Set of columns of type [[GridColumns]]. */ @@ -395,10 +393,6 @@ interface GridComponentOtherProps { * If one of the data in `initialState` is also being controlled, then the control state wins. */ initialState?: GridInitialState; - /** - * @ignore - do not document - */ - style?: React.CSSProperties; /** * Overrideable components props dynamically passed to the component at rendering. */ @@ -409,4 +403,8 @@ interface GridComponentOtherProps { groupingColDef?: | GridColDefOverride<'field' | 'editable'> | GridColDefOverrideCallback<'field' | 'editable'>; + /** + * The system prop that allows defining system overrides as well as additional CSS styles. + */ + sx?: SxProps; } diff --git a/packages/grid/_modules_/grid/components/containers/GridRoot.tsx b/packages/grid/_modules_/grid/components/containers/GridRoot.tsx index 5c636a65df3d6..44f68b5951489 100644 --- a/packages/grid/_modules_/grid/components/containers/GridRoot.tsx +++ b/packages/grid/_modules_/grid/components/containers/GridRoot.tsx @@ -1,6 +1,9 @@ import * as React from 'react'; +import PropTypes from 'prop-types'; import clsx from 'clsx'; import { useForkRef } from '@mui/material/utils'; +import { SxProps } from '@mui/system'; +import { Theme } from '@mui/material/styles'; import NoSsr from '@mui/material/NoSsr'; import { GridRootContainerRef } from '../../models/gridRootContainerRef'; import { GridRootStyles } from './GridRootStyles'; @@ -11,12 +14,14 @@ import { useGridRootProps } from '../../hooks/utils/useGridRootProps'; import { gridClasses } from '../../gridClasses'; import { gridRowCountSelector } from '../../hooks/features/rows/gridRowsSelector'; -export type GridRootProps = React.HTMLAttributes; +export interface GridRootProps extends React.HTMLAttributes { + /** + * The system prop that allows defining system overrides as well as additional CSS styles. + */ + sx?: SxProps; +} -export const GridRoot = React.forwardRef(function GridRoot( - props, - ref, -) { +const GridRoot = React.forwardRef(function GridRoot(props, ref) { const rootProps = useGridRootProps(); const { children, className, ...other } = props; const apiRef = useGridApiContext(); @@ -31,7 +36,7 @@ export const GridRoot = React.forwardRef(function (function aria-multiselectable={!rootProps.disableMultipleSelection} aria-label={rootProps['aria-label']} aria-labelledby={rootProps['aria-labelledby']} - style={rootProps.style} {...other} > {children} @@ -48,3 +52,20 @@ export const GridRoot = React.forwardRef(function ); }); + +GridRoot.propTypes = { + // ----------------------------- Warning -------------------------------- + // | These PropTypes are generated from the TypeScript type definitions | + // | To update them edit the TypeScript types and run "yarn proptypes" | + // ---------------------------------------------------------------------- + /** + * The system prop that allows defining system overrides as well as additional CSS styles. + */ + sx: PropTypes.oneOfType([ + PropTypes.arrayOf(PropTypes.oneOfType([PropTypes.func, PropTypes.object])), + PropTypes.func, + PropTypes.object, + ]), +} as any; + +export { GridRoot }; diff --git a/packages/grid/x-data-grid-pro/package.json b/packages/grid/x-data-grid-pro/package.json index c18eacbfd3e6b..77deef9d0114c 100644 --- a/packages/grid/x-data-grid-pro/package.json +++ b/packages/grid/x-data-grid-pro/package.json @@ -45,6 +45,7 @@ }, "peerDependencies": { "@mui/material": "^5.1.1", + "@mui/system": "^5.1.1", "react": "^17.0.2" }, "setupFiles": [ diff --git a/packages/grid/x-data-grid-pro/src/DataGridPro.tsx b/packages/grid/x-data-grid-pro/src/DataGridPro.tsx index 6890a0e8926da..38dd860fe82b3 100644 --- a/packages/grid/x-data-grid-pro/src/DataGridPro.tsx +++ b/packages/grid/x-data-grid-pro/src/DataGridPro.tsx @@ -40,7 +40,7 @@ const DataGridProRaw = React.forwardRef(functi return ( - + ; +} diff --git a/packages/grid/x-data-grid-pro/src/tests/layout.DataGridPro.test.tsx b/packages/grid/x-data-grid-pro/src/tests/layout.DataGridPro.test.tsx index f9c63c90eb293..d4fef257ee55e 100644 --- a/packages/grid/x-data-grid-pro/src/tests/layout.DataGridPro.test.tsx +++ b/packages/grid/x-data-grid-pro/src/tests/layout.DataGridPro.test.tsx @@ -1,5 +1,5 @@ import * as React from 'react'; -import { createRenderer } from '@material-ui/monorepo/test/utils'; +import { createRenderer, screen } from '@material-ui/monorepo/test/utils'; import { expect } from 'chai'; import { createTheme, ThemeProvider } from '@mui/material/styles'; import { GridApiRef, useGridApiRef, DataGridPro, ptBR } from '@mui/x-data-grid-pro'; @@ -147,4 +147,27 @@ describe(' - Layout', () => { ); expect(document.querySelector('[title="Ordenar"]')).not.to.equal(null); }); + + it('should support the sx prop', () => { + const theme = createTheme({ + palette: { + primary: { + main: 'rgb(0, 0, 255)', + }, + }, + }); + + render( + +
+ +
+
, + ); + + // @ts-expect-error need to migrate helpers to TypeScript + expect(screen.getByRole('grid')).toHaveComputedStyle({ + color: 'rgb(0, 0, 255)', + }); + }); }); diff --git a/packages/grid/x-data-grid/package.json b/packages/grid/x-data-grid/package.json index c7212c6d558c9..65234086906a9 100644 --- a/packages/grid/x-data-grid/package.json +++ b/packages/grid/x-data-grid/package.json @@ -44,6 +44,7 @@ }, "peerDependencies": { "@mui/material": "^5.1.1", + "@mui/system": "^5.1.1", "react": "^17.0.2" }, "setupFiles": [ diff --git a/packages/grid/x-data-grid/src/DataGrid.tsx b/packages/grid/x-data-grid/src/DataGrid.tsx index 72d505ec606b7..2b841383db1e9 100644 --- a/packages/grid/x-data-grid/src/DataGrid.tsx +++ b/packages/grid/x-data-grid/src/DataGrid.tsx @@ -26,7 +26,7 @@ const DataGridRaw = React.forwardRef(function Dat return ( - + ); } + +function SxTest() { + ; +} diff --git a/packages/grid/x-data-grid/src/tests/layout.DataGrid.test.tsx b/packages/grid/x-data-grid/src/tests/layout.DataGrid.test.tsx index 2f18e153ad2d8..a8c0aa58ffd71 100644 --- a/packages/grid/x-data-grid/src/tests/layout.DataGrid.test.tsx +++ b/packages/grid/x-data-grid/src/tests/layout.DataGrid.test.tsx @@ -812,4 +812,27 @@ describe(' - Layout & Warnings', () => { expect(window.getComputedStyle(getRow(0)).backgroundColor).to.equal('rgb(128, 0, 128)'); expect(window.getComputedStyle(getCell(0, 0)).backgroundColor).to.equal('rgb(0, 128, 0)'); }); + + it('should support the sx prop', () => { + const theme = createTheme({ + palette: { + primary: { + main: 'rgb(0, 0, 255)', + }, + }, + }); + + render( + +
+ +
+
, + ); + + // @ts-expect-error need to migrate helpers to TypeScript + expect(screen.getByRole('grid')).toHaveComputedStyle({ + color: 'rgb(0, 0, 255)', + }); + }); }); diff --git a/scripts/exportsSnapshot.json b/scripts/exportsSnapshot.json index 418c20e9b4f79..30645b41d1652 100644 --- a/scripts/exportsSnapshot.json +++ b/scripts/exportsSnapshot.json @@ -114,6 +114,7 @@ { "name": "GridRenderEditCellParams", "kind": "Interface" }, { "name": "GridRenderPaginationProps", "kind": "Interface" }, { "name": "GridRenderRowProps", "kind": "Interface" }, + { "name": "GridRootProps", "kind": "Interface" }, { "name": "GridRowApi", "kind": "Interface" }, { "name": "GridRowEventLookup", "kind": "Interface" }, { "name": "GridRowModelUpdate", "kind": "Interface" }, @@ -188,7 +189,6 @@ { "name": "GridPreferencePanelInitialState", "kind": "Type alias" }, { "name": "GridRenderContextProps", "kind": "Type alias" }, { "name": "GridRootContainerRef", "kind": "Type alias" }, - { "name": "GridRootProps", "kind": "Type alias" }, { "name": "GridRowData", "kind": "Type alias" }, { "name": "GridRowEntry", "kind": "Type alias" }, { "name": "GridRowId", "kind": "Type alias" },