diff --git a/docs/data/data-grid/custom-columns/CustomColumnFullExample.js b/docs/data/data-grid/custom-columns/CustomColumnFullExample.js new file mode 100644 index 0000000000000..be7c002fb2b74 --- /dev/null +++ b/docs/data/data-grid/custom-columns/CustomColumnFullExample.js @@ -0,0 +1,159 @@ +import * as React from 'react'; +import { + randomColor, + randomEmail, + randomInt, + randomName, + randomArrayItem, + random, +} from '@mui/x-data-grid-generator'; +import { DataGrid, gridStringOrNumberComparator } from '@mui/x-data-grid'; +import { renderAvatar } from './cell-renderers/avatar'; +import { renderEmail } from './cell-renderers/email'; +import { renderEditRating, renderRating } from './cell-renderers/rating'; +import { + COUNTRY_ISO_OPTIONS, + renderCountry, + renderEditCountry, +} from './cell-renderers/country'; +import { renderSparkline } from './cell-renderers/sparkline'; +import { renderEditProgress, renderProgress } from './cell-renderers/progress'; +import { + renderEditStatus, + renderStatus, + STATUS_OPTIONS, +} from './cell-renderers/status'; +import { + INCOTERM_OPTIONS, + renderEditIncoterm, + renderIncoterm, +} from './cell-renderers/incoterm'; + +const columns = [ + { + field: 'name', + headerName: 'Name', + width: 120, + editable: true, + }, + { + field: 'avatar', + headerName: 'Avatar', + display: 'flex', + renderCell: renderAvatar, + valueGetter: (value, row) => + row.name == null || row.avatar == null + ? null + : { name: row.name, color: row.avatar }, + sortable: false, + filterable: false, + }, + { + field: 'email', + headerName: 'Email', + renderCell: renderEmail, + width: 150, + editable: true, + }, + { + field: 'rating', + headerName: 'Rating', + display: 'flex', + renderCell: renderRating, + renderEditCell: renderEditRating, + width: 180, + type: 'number', + editable: true, + availableAggregationFunctions: ['avg', 'min', 'max', 'size'], + }, + { + field: 'country', + headerName: 'Country', + type: 'singleSelect', + valueOptions: COUNTRY_ISO_OPTIONS, + valueFormatter: (value) => value?.label, + renderCell: renderCountry, + renderEditCell: renderEditCountry, + sortComparator: (v1, v2, param1, param2) => + gridStringOrNumberComparator(v1.label, v2.label, param1, param2), + width: 150, + editable: true, + }, + { + field: 'salary', + headerName: 'Salary', + type: 'number', + valueFormatter: (value) => { + if (!value || typeof value !== 'number') { + return value; + } + return `$${value.toLocaleString()}`; + }, + editable: true, + }, + { + field: 'monthlyActivity', + headerName: 'Monthly activity', + type: 'custom', + resizable: false, + filterable: false, + sortable: false, + editable: false, + groupable: false, + display: 'flex', + renderCell: renderSparkline, + width: 150, + valueGetter: (value, row) => row.monthlyActivity, + }, + { + field: 'budget', + headerName: 'Budget left', + renderCell: renderProgress, + renderEditCell: renderEditProgress, + availableAggregationFunctions: ['min', 'max', 'avg', 'size'], + type: 'number', + width: 120, + editable: true, + }, + { + field: 'status', + headerName: 'Status', + renderCell: renderStatus, + renderEditCell: renderEditStatus, + type: 'singleSelect', + valueOptions: STATUS_OPTIONS, + width: 150, + editable: true, + }, + { + field: 'incoTerm', + headerName: 'Incoterm', + renderCell: renderIncoterm, + renderEditCell: renderEditIncoterm, + type: 'singleSelect', + valueOptions: INCOTERM_OPTIONS, + editable: true, + }, +]; + +const rows = Array.from({ length: 10 }, (_, index) => ({ + id: index, + name: randomName({}, {}), + avatar: randomColor(), + email: randomEmail(), + rating: randomInt(1, 5), + country: randomArrayItem(COUNTRY_ISO_OPTIONS), + salary: randomInt(35000, 80000), + monthlyActivity: Array.from({ length: 30 }, () => randomInt(1, 25)), + budget: random(0, 1).toPrecision(), + status: randomArrayItem(STATUS_OPTIONS), + incoTerm: randomArrayItem(INCOTERM_OPTIONS), +})); + +export default function CustomColumnFullExample() { + return ( +
+ +
+ ); +} diff --git a/docs/data/data-grid/custom-columns/CustomColumnFullExample.tsx b/docs/data/data-grid/custom-columns/CustomColumnFullExample.tsx new file mode 100644 index 0000000000000..2f0fa78b81a4c --- /dev/null +++ b/docs/data/data-grid/custom-columns/CustomColumnFullExample.tsx @@ -0,0 +1,164 @@ +import * as React from 'react'; +import { + randomColor, + randomEmail, + randomInt, + randomName, + randomArrayItem, + random, +} from '@mui/x-data-grid-generator'; +import { + DataGrid, + GridColDef, + gridStringOrNumberComparator, +} from '@mui/x-data-grid'; +import { renderAvatar } from './cell-renderers/avatar'; +import { renderEmail } from './cell-renderers/email'; +import { renderEditRating, renderRating } from './cell-renderers/rating'; +import { + COUNTRY_ISO_OPTIONS, + CountryIsoOption, + renderCountry, + renderEditCountry, +} from './cell-renderers/country'; +import { renderSparkline } from './cell-renderers/sparkline'; +import { renderEditProgress, renderProgress } from './cell-renderers/progress'; +import { + renderEditStatus, + renderStatus, + STATUS_OPTIONS, +} from './cell-renderers/status'; +import { + INCOTERM_OPTIONS, + renderEditIncoterm, + renderIncoterm, +} from './cell-renderers/incoterm'; + +const columns: GridColDef<(typeof rows)[number]>[] = [ + { + field: 'name', + headerName: 'Name', + width: 120, + editable: true, + }, + { + field: 'avatar', + headerName: 'Avatar', + display: 'flex', + renderCell: renderAvatar, + valueGetter: (value, row) => + row.name == null || row.avatar == null + ? null + : { name: row.name, color: row.avatar }, + sortable: false, + filterable: false, + } as GridColDef, + { + field: 'email', + headerName: 'Email', + renderCell: renderEmail, + width: 150, + editable: true, + }, + { + field: 'rating', + headerName: 'Rating', + display: 'flex', + renderCell: renderRating, + renderEditCell: renderEditRating, + width: 180, + type: 'number', + editable: true, + availableAggregationFunctions: ['avg', 'min', 'max', 'size'], + }, + { + field: 'country', + headerName: 'Country', + type: 'singleSelect', + valueOptions: COUNTRY_ISO_OPTIONS, + valueFormatter: (value: CountryIsoOption) => value?.label, + renderCell: renderCountry, + renderEditCell: renderEditCountry, + sortComparator: (v1, v2, param1, param2) => + gridStringOrNumberComparator(v1.label, v2.label, param1, param2), + width: 150, + editable: true, + } as GridColDef, + { + field: 'salary', + headerName: 'Salary', + type: 'number', + valueFormatter: (value?: number) => { + if (!value || typeof value !== 'number') { + return value; + } + return `$${value.toLocaleString()}`; + }, + editable: true, + }, + { + field: 'monthlyActivity', + headerName: 'Monthly activity', + type: 'custom', + resizable: false, + filterable: false, + sortable: false, + editable: false, + groupable: false, + display: 'flex', + renderCell: renderSparkline, + width: 150, + valueGetter: (value, row) => row.monthlyActivity, + }, + { + field: 'budget', + headerName: 'Budget left', + renderCell: renderProgress, + renderEditCell: renderEditProgress, + availableAggregationFunctions: ['min', 'max', 'avg', 'size'], + type: 'number', + width: 120, + editable: true, + }, + { + field: 'status', + headerName: 'Status', + renderCell: renderStatus, + renderEditCell: renderEditStatus, + type: 'singleSelect', + valueOptions: STATUS_OPTIONS, + width: 150, + editable: true, + }, + { + field: 'incoTerm', + headerName: 'Incoterm', + renderCell: renderIncoterm, + renderEditCell: renderEditIncoterm, + type: 'singleSelect', + valueOptions: INCOTERM_OPTIONS, + editable: true, + }, +]; + +const rows = Array.from({ length: 10 }, (_, index) => ({ + id: index, + name: randomName({}, {}), + avatar: randomColor(), + email: randomEmail(), + rating: randomInt(1, 5), + country: randomArrayItem(COUNTRY_ISO_OPTIONS), + salary: randomInt(35000, 80000), + monthlyActivity: Array.from({ length: 30 }, () => randomInt(1, 25)), + budget: random(0, 1).toPrecision(), + status: randomArrayItem(STATUS_OPTIONS), + incoTerm: randomArrayItem(INCOTERM_OPTIONS), +})); + +export default function CustomColumnFullExample() { + return ( +
+ +
+ ); +} diff --git a/docs/data/data-grid/custom-columns/CustomColumnFullExample.tsx.preview b/docs/data/data-grid/custom-columns/CustomColumnFullExample.tsx.preview new file mode 100644 index 0000000000000..074afd47c4411 --- /dev/null +++ b/docs/data/data-grid/custom-columns/CustomColumnFullExample.tsx.preview @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/docs/data/data-grid/custom-columns/cell-renderers/avatar.js b/docs/data/data-grid/custom-columns/cell-renderers/avatar.js new file mode 100644 index 0000000000000..a928435e763ce --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/avatar.js @@ -0,0 +1,14 @@ +import * as React from 'react'; +import Avatar from '@mui/material/Avatar'; + +export function renderAvatar(params) { + if (params.value == null) { + return ''; + } + + return ( + + {params.value.name.toUpperCase().substring(0, 1)} + + ); +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/avatar.tsx b/docs/data/data-grid/custom-columns/cell-renderers/avatar.tsx new file mode 100644 index 0000000000000..e2f1f5b5b1a99 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/avatar.tsx @@ -0,0 +1,17 @@ +import * as React from 'react'; +import Avatar from '@mui/material/Avatar'; +import { GridRenderCellParams } from '@mui/x-data-grid'; + +export function renderAvatar( + params: GridRenderCellParams<{ name: string; color: string }, any, any>, +) { + if (params.value == null) { + return ''; + } + + return ( + + {params.value.name.toUpperCase().substring(0, 1)} + + ); +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/country.js b/docs/data/data-grid/custom-columns/cell-renderers/country.js new file mode 100644 index 0000000000000..d8dbd277fb67e --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/country.js @@ -0,0 +1,130 @@ +import * as React from 'react'; +import { useGridApiContext } from '@mui/x-data-grid'; +import Autocomplete, { autocompleteClasses } from '@mui/material/Autocomplete'; + +import Box from '@mui/material/Box'; +import InputBase from '@mui/material/InputBase'; +import { styled } from '@mui/material/styles'; + +export const COUNTRY_ISO_OPTIONS = [ + { value: 'DE', code: 'DE', label: 'Germany', phone: '49' }, + { value: 'ES', code: 'ES', label: 'Spain', phone: '34' }, + { value: 'FR', code: 'FR', label: 'France', phone: '33' }, + { value: 'GB', code: 'GB', label: 'United Kingdom', phone: '44' }, +]; + +const Country = React.memo(function Country(props) { + const { value } = props; + + return ( + img': { + mr: 0.5, + flexShrink: 0, + width: '20px', + }, + }} + > + + + {value.label} + + + ); +}); + +const StyledAutocomplete = styled(Autocomplete)(({ theme }) => ({ + height: '100%', + [`& .${autocompleteClasses.inputRoot}`]: { + ...theme.typography.body2, + padding: '1px 0', + height: '100%', + '& input': { + padding: '0 16px', + height: '100%', + }, + }, +})); + +function EditCountry(props) { + const { id, value, field } = props; + + const apiRef = useGridApiContext(); + + const handleChange = React.useCallback( + async (event, newValue) => { + await apiRef.current.setEditCellValue({ id, field, value: newValue }, event); + apiRef.current.stopCellEditMode({ id, field }); + }, + [apiRef, field, id], + ); + + return ( + option.label} + autoHighlight + fullWidth + open + disableClearable + renderOption={(optionProps, option) => ( + img': { + mr: 1.5, + flexShrink: 0, + }, + }} + {...optionProps} + key={option.code} + > + + {option.label} + + )} + renderInput={(params) => ( + + )} + /> + ); +} + +export function renderCountry(params) { + if (params.value == null) { + return ''; + } + + return ; +} + +export function renderEditCountry(params) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/country.tsx b/docs/data/data-grid/custom-columns/cell-renderers/country.tsx new file mode 100644 index 0000000000000..54a3a293aac9f --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/country.tsx @@ -0,0 +1,152 @@ +import * as React from 'react'; +import { + GridRenderCellParams, + GridRenderEditCellParams, + useGridApiContext, +} from '@mui/x-data-grid'; +import Autocomplete, { autocompleteClasses } from '@mui/material/Autocomplete'; +import type { AutocompleteProps } from '@mui/material/Autocomplete'; +import Box from '@mui/material/Box'; +import InputBase from '@mui/material/InputBase'; +import { styled } from '@mui/material/styles'; + +export interface CountryIsoOption { + value: string; + code: string; + label: string; + phone: string; + suggested?: boolean; +} + +export const COUNTRY_ISO_OPTIONS: CountryIsoOption[] = [ + { value: 'DE', code: 'DE', label: 'Germany', phone: '49' }, + { value: 'ES', code: 'ES', label: 'Spain', phone: '34' }, + { value: 'FR', code: 'FR', label: 'France', phone: '33' }, + { value: 'GB', code: 'GB', label: 'United Kingdom', phone: '44' }, +]; + +interface CountryProps { + value: CountryIsoOption; +} + +const Country = React.memo(function Country(props: CountryProps) { + const { value } = props; + + return ( + img': { + mr: 0.5, + flexShrink: 0, + width: '20px', + }, + }} + > + + + {value.label} + + + ); +}); + +const StyledAutocomplete = styled(Autocomplete)(({ theme }) => ({ + height: '100%', + [`& .${autocompleteClasses.inputRoot}`]: { + ...theme.typography.body2, + padding: '1px 0', + height: '100%', + '& input': { + padding: '0 16px', + height: '100%', + }, + }, +})) as typeof Autocomplete; + +function EditCountry(props: GridRenderEditCellParams) { + const { id, value, field } = props; + + const apiRef = useGridApiContext(); + + const handleChange = React.useCallback< + NonNullable['onChange']> + >( + async (event, newValue) => { + await apiRef.current.setEditCellValue({ id, field, value: newValue }, event); + apiRef.current.stopCellEditMode({ id, field }); + }, + [apiRef, field, id], + ); + + return ( + + value={value} + onChange={handleChange} + options={COUNTRY_ISO_OPTIONS} + getOptionLabel={(option: any) => option.label} + autoHighlight + fullWidth + open + disableClearable + renderOption={(optionProps, option: any) => ( + img': { + mr: 1.5, + flexShrink: 0, + }, + }} + {...optionProps} + key={option.code} + > + + {option.label} + + )} + renderInput={(params) => ( + + )} + /> + ); +} + +export function renderCountry( + params: GridRenderCellParams, +) { + if (params.value == null) { + return ''; + } + + return ; +} + +export function renderEditCountry( + params: GridRenderEditCellParams, +) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/email.js b/docs/data/data-grid/custom-columns/cell-renderers/email.js new file mode 100644 index 0000000000000..a80f6c387fba8 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/email.js @@ -0,0 +1,32 @@ +import * as React from 'react'; +import { styled } from '@mui/material/styles'; + +const Link = styled('a')({ + textOverflow: 'ellipsis', + whiteSpace: 'nowrap', + overflow: 'hidden', + color: 'inherit', +}); + +const DemoLink = React.memo(function DemoLink(props) { + const handleClick = (event) => { + event.preventDefault(); + event.stopPropagation(); + }; + + return ( + + {props.children} + + ); +}); + +export function renderEmail(params) { + const email = params.value ?? ''; + + return ( + + {email} + + ); +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/email.tsx b/docs/data/data-grid/custom-columns/cell-renderers/email.tsx new file mode 100644 index 0000000000000..e58fc5d782614 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/email.tsx @@ -0,0 +1,39 @@ +import * as React from 'react'; +import { styled } from '@mui/material/styles'; +import { GridRenderCellParams } from '@mui/x-data-grid'; + +interface DemoLinkProps { + href: string; + children: string; + tabIndex: number; +} + +const Link = styled('a')({ + textOverflow: 'ellipsis', + whiteSpace: 'nowrap', + overflow: 'hidden', + color: 'inherit', +}); + +const DemoLink = React.memo(function DemoLink(props: DemoLinkProps) { + const handleClick = (event: React.MouseEvent) => { + event.preventDefault(); + event.stopPropagation(); + }; + + return ( + + {props.children} + + ); +}); + +export function renderEmail(params: GridRenderCellParams) { + const email = params.value ?? ''; + + return ( + + {email} + + ); +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/incoterm.js b/docs/data/data-grid/custom-columns/cell-renderers/incoterm.js new file mode 100644 index 0000000000000..524293d927540 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/incoterm.js @@ -0,0 +1,104 @@ +import * as React from 'react'; +import Box from '@mui/material/Box'; +import ListItemIcon from '@mui/material/ListItemIcon'; +import ListItemText from '@mui/material/ListItemText'; +import MenuItem from '@mui/material/MenuItem'; + +import Select from '@mui/material/Select'; +import Tooltip from '@mui/material/Tooltip'; +import InfoIcon from '@mui/icons-material/Info'; +import { useGridApiContext } from '@mui/x-data-grid'; + +export const INCOTERM_OPTIONS = [ + 'EXW (Ex Works)', + 'FAS (Free Alongside Ship)', + 'FCA (Free Carrier)', + 'CPT (Carriage Paid To)', + 'DAP (Delivered at Place)', + 'DPU (Delivered at Place Unloaded)', + 'DDP (Delivered Duty Paid)', +]; + +const Incoterm = React.memo(function Incoterm(props) { + const { value } = props; + + if (!value) { + return null; + } + + const valueStr = value.toString(); + const tooltip = valueStr.slice(valueStr.indexOf('(') + 1, valueStr.indexOf(')')); + const code = valueStr.slice(0, valueStr.indexOf('(')).trim(); + + return ( + + {code} + + + + + ); +}); + +function EditIncoterm(props) { + const { id, value, field } = props; + + const apiRef = useGridApiContext(); + + const handleChange = async (event) => { + await apiRef.current.setEditCellValue( + { id, field, value: event.target.value }, + event, + ); + apiRef.current.stopCellEditMode({ id, field }); + }; + + const handleClose = (event, reason) => { + if (reason === 'backdropClick') { + apiRef.current.stopCellEditMode({ id, field }); + } + }; + + return ( + + ); +} + +export function renderIncoterm(params) { + return ; +} + +export function renderEditIncoterm(params) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/incoterm.tsx b/docs/data/data-grid/custom-columns/cell-renderers/incoterm.tsx new file mode 100644 index 0000000000000..de90756039bfd --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/incoterm.tsx @@ -0,0 +1,116 @@ +import * as React from 'react'; +import Box from '@mui/material/Box'; +import ListItemIcon from '@mui/material/ListItemIcon'; +import ListItemText from '@mui/material/ListItemText'; +import MenuItem from '@mui/material/MenuItem'; +import type { MenuProps } from '@mui/material/Menu'; +import Select, { SelectProps } from '@mui/material/Select'; +import Tooltip from '@mui/material/Tooltip'; +import InfoIcon from '@mui/icons-material/Info'; +import { + GridRenderCellParams, + GridRenderEditCellParams, + useGridApiContext, +} from '@mui/x-data-grid'; + +export const INCOTERM_OPTIONS = [ + 'EXW (Ex Works)', + 'FAS (Free Alongside Ship)', + 'FCA (Free Carrier)', + 'CPT (Carriage Paid To)', + 'DAP (Delivered at Place)', + 'DPU (Delivered at Place Unloaded)', + 'DDP (Delivered Duty Paid)', +]; + +interface IncotermProps { + value: string | null | undefined; +} + +const Incoterm = React.memo(function Incoterm(props: IncotermProps) { + const { value } = props; + + if (!value) { + return null; + } + + const valueStr = value.toString(); + const tooltip = valueStr.slice(valueStr.indexOf('(') + 1, valueStr.indexOf(')')); + const code = valueStr.slice(0, valueStr.indexOf('(')).trim(); + + return ( + + {code} + + + + + ); +}); + +function EditIncoterm(props: GridRenderEditCellParams) { + const { id, value, field } = props; + + const apiRef = useGridApiContext(); + + const handleChange: SelectProps['onChange'] = async (event) => { + await apiRef.current.setEditCellValue( + { id, field, value: event.target.value as any }, + event, + ); + apiRef.current.stopCellEditMode({ id, field }); + }; + + const handleClose: MenuProps['onClose'] = (event, reason) => { + if (reason === 'backdropClick') { + apiRef.current.stopCellEditMode({ id, field }); + } + }; + + return ( + + ); +} + +export function renderIncoterm( + params: GridRenderCellParams, +) { + return ; +} + +export function renderEditIncoterm( + params: GridRenderEditCellParams, +) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/progress.js b/docs/data/data-grid/custom-columns/cell-renderers/progress.js new file mode 100644 index 0000000000000..d604aaca719af --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/progress.js @@ -0,0 +1,178 @@ +import * as React from 'react'; +import clsx from 'clsx'; +import { useGridApiContext } from '@mui/x-data-grid'; +import { alpha, styled } from '@mui/material/styles'; +import Slider, { sliderClasses } from '@mui/material/Slider'; +import Tooltip from '@mui/material/Tooltip'; +import { debounce } from '@mui/material/utils'; + +const Center = styled('div')({ + height: '100%', + display: 'flex', + alignItems: 'center', +}); + +const Element = styled('div')(({ theme }) => ({ + border: `1px solid ${(theme.vars || theme).palette.divider}`, + position: 'relative', + overflow: 'hidden', + width: '100%', + height: 26, + borderRadius: 2, +})); + +const Value = styled('div')({ + position: 'absolute', + lineHeight: '24px', + width: '100%', + display: 'flex', + justifyContent: 'center', +}); + +const Bar = styled('div')({ + height: '100%', + '&.low': { + backgroundColor: '#f44336', + }, + '&.medium': { + backgroundColor: '#efbb5aa3', + }, + '&.high': { + backgroundColor: '#088208a3', + }, +}); + +const ProgressBar = React.memo(function ProgressBar(props) { + const { value } = props; + const valueInPercent = value * 100; + + return ( + + {`${valueInPercent.toLocaleString()} %`} + = 30 && valueInPercent <= 70, + high: valueInPercent > 70, + })} + style={{ maxWidth: `${valueInPercent}%` }} + /> + + ); +}); + +const StyledSlider = styled(Slider)(({ theme }) => ({ + display: 'flex', + height: '100%', + width: '100%', + alignItems: 'center', + justifyContent: 'center', + padding: 0, + borderRadius: 0, + [`& .${sliderClasses.rail}`]: { + height: '100%', + backgroundColor: 'transparent', + }, + [`& .${sliderClasses.track}`]: { + height: '100%', + transition: theme.transitions.create('background-color', { + duration: theme.transitions.duration.shorter, + }), + '&.low': { + backgroundColor: '#f44336', + }, + '&.medium': { + backgroundColor: '#efbb5aa3', + }, + '&.high': { + backgroundColor: '#088208a3', + }, + }, + [`& .${sliderClasses.thumb}`]: { + height: '100%', + width: 5, + borderRadius: 0, + marginTop: 0, + backgroundColor: alpha('#000000', 0.2), + }, +})); + +const ValueLabelComponent = React.memo(function ValueLabelComponent(props) { + const { children, open, value } = props; + + return ( + + {children} + + ); +}); + +function EditProgress(props) { + const { id, value, field } = props; + const [valueState, setValueState] = React.useState(Number(value)); + + const apiRef = useGridApiContext(); + + const updateCellEditProps = React.useCallback( + (newValue) => { + apiRef.current.setEditCellValue({ id, field, value: newValue }); + }, + [apiRef, field, id], + ); + + const debouncedUpdateCellEditProps = React.useMemo( + () => debounce(updateCellEditProps, 60), + [updateCellEditProps], + ); + + const handleChange = (event, newValue) => { + setValueState(newValue); + debouncedUpdateCellEditProps(newValue); + }; + + React.useEffect(() => { + setValueState(Number(value)); + }, [value]); + + const handleRef = (element) => { + if (element) { + element.querySelector('[type="range"]').focus(); + } + }; + + return ( + = 0.3 && valueState <= 0.7, + high: valueState > 0.7, + }), + }} + value={valueState} + max={1} + step={0.00001} + onChange={handleChange} + components={{ ValueLabel: ValueLabelComponent }} + valueLabelDisplay="auto" + valueLabelFormat={(newValue) => `${(newValue * 100).toLocaleString()} %`} + /> + ); +} + +export function renderProgress(params) { + if (params.value == null) { + return ''; + } + + return ( +
+ +
+ ); +} + +export function renderEditProgress(params) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/progress.tsx b/docs/data/data-grid/custom-columns/cell-renderers/progress.tsx new file mode 100644 index 0000000000000..a831d003ff58e --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/progress.tsx @@ -0,0 +1,186 @@ +import * as React from 'react'; +import clsx from 'clsx'; +import { + GridRenderCellParams, + GridRenderEditCellParams, + useGridApiContext, +} from '@mui/x-data-grid'; +import { alpha, styled } from '@mui/material/styles'; +import Slider, { sliderClasses, SliderProps } from '@mui/material/Slider'; +import Tooltip from '@mui/material/Tooltip'; +import { debounce } from '@mui/material/utils'; + +interface ProgressBarProps { + value: number; +} + +const Center = styled('div')({ + height: '100%', + display: 'flex', + alignItems: 'center', +}); + +const Element = styled('div')(({ theme }) => ({ + border: `1px solid ${(theme.vars || theme).palette.divider}`, + position: 'relative', + overflow: 'hidden', + width: '100%', + height: 26, + borderRadius: 2, +})); + +const Value = styled('div')({ + position: 'absolute', + lineHeight: '24px', + width: '100%', + display: 'flex', + justifyContent: 'center', +}); + +const Bar = styled('div')({ + height: '100%', + '&.low': { + backgroundColor: '#f44336', + }, + '&.medium': { + backgroundColor: '#efbb5aa3', + }, + '&.high': { + backgroundColor: '#088208a3', + }, +}); + +const ProgressBar = React.memo(function ProgressBar(props: ProgressBarProps) { + const { value } = props; + const valueInPercent = value * 100; + + return ( + + {`${valueInPercent.toLocaleString()} %`} + = 30 && valueInPercent <= 70, + high: valueInPercent > 70, + })} + style={{ maxWidth: `${valueInPercent}%` }} + /> + + ); +}); + +const StyledSlider = styled(Slider)(({ theme }) => ({ + display: 'flex', + height: '100%', + width: '100%', + alignItems: 'center', + justifyContent: 'center', + padding: 0, + borderRadius: 0, + [`& .${sliderClasses.rail}`]: { + height: '100%', + backgroundColor: 'transparent', + }, + [`& .${sliderClasses.track}`]: { + height: '100%', + transition: theme.transitions.create('background-color', { + duration: theme.transitions.duration.shorter, + }), + '&.low': { + backgroundColor: '#f44336', + }, + '&.medium': { + backgroundColor: '#efbb5aa3', + }, + '&.high': { + backgroundColor: '#088208a3', + }, + }, + [`& .${sliderClasses.thumb}`]: { + height: '100%', + width: 5, + borderRadius: 0, + marginTop: 0, + backgroundColor: alpha('#000000', 0.2), + }, +})); + +const ValueLabelComponent = React.memo(function ValueLabelComponent(props: any) { + const { children, open, value } = props; + + return ( + + {children} + + ); +}); + +function EditProgress(props: GridRenderEditCellParams) { + const { id, value, field } = props; + const [valueState, setValueState] = React.useState(Number(value)); + + const apiRef = useGridApiContext(); + + const updateCellEditProps = React.useCallback( + (newValue: number) => { + apiRef.current.setEditCellValue({ id, field, value: newValue }); + }, + [apiRef, field, id], + ); + + const debouncedUpdateCellEditProps = React.useMemo( + () => debounce(updateCellEditProps, 60), + [updateCellEditProps], + ); + + const handleChange = (event: Event, newValue: number | number[]) => { + setValueState(newValue as number); + debouncedUpdateCellEditProps(newValue as number); + }; + + React.useEffect(() => { + setValueState(Number(value)); + }, [value]); + + const handleRef: SliderProps['ref'] = (element) => { + if (element) { + element.querySelector('[type="range"]')!.focus(); + } + }; + + return ( + = 0.3 && valueState <= 0.7, + high: valueState > 0.7, + }), + }} + value={valueState} + max={1} + step={0.00001} + onChange={handleChange} + components={{ ValueLabel: ValueLabelComponent }} + valueLabelDisplay="auto" + valueLabelFormat={(newValue) => `${(newValue * 100).toLocaleString()} %`} + /> + ); +} + +export function renderProgress(params: GridRenderCellParams) { + if (params.value == null) { + return ''; + } + + return ( +
+ +
+ ); +} + +export function renderEditProgress(params: GridRenderEditCellParams) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/rating.js b/docs/data/data-grid/custom-columns/cell-renderers/rating.js new file mode 100644 index 0000000000000..46d1d09ea7516 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/rating.js @@ -0,0 +1,93 @@ +import * as React from 'react'; +import Box from '@mui/material/Box'; +import Rating from '@mui/material/Rating'; +import { useGridApiContext } from '@mui/x-data-grid'; + +const RatingValue = React.memo(function RatingValue(props) { + const { value } = props; + return ( + + {' '} + {Math.round(Number(value) * 10) / 10} + + ); +}); + +function EditRating(props) { + const { id, value, field } = props; + + const apiRef = useGridApiContext(); + + const changedThroughKeyboard = React.useRef(false); + + const handleChange = async (event) => { + await apiRef.current.setEditCellValue( + { id, field, value: Number(event.target.value) }, + event, + ); + if (!changedThroughKeyboard.current) { + apiRef.current.stopCellEditMode({ id, field }); + } + changedThroughKeyboard.current = false; + }; + + const handleRef = (element) => { + if (element) { + if (value !== 0) { + element.querySelector(`input[value="${value}"]`).focus(); + } else { + element.querySelector('input[value=""]').focus(); + } + } + }; + + const handleKeyDown = (event) => { + if (event.key.startsWith('Arrow')) { + changedThroughKeyboard.current = true; + } else { + changedThroughKeyboard.current = false; + } + }; + + return ( + + + {Number(value)} + + ); +} + +export function renderRating(params) { + if (params.value == null) { + return ''; + } + + return ; +} + +export function renderEditRating(params) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/rating.tsx b/docs/data/data-grid/custom-columns/cell-renderers/rating.tsx new file mode 100644 index 0000000000000..9eb9c68ce767d --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/rating.tsx @@ -0,0 +1,101 @@ +import * as React from 'react'; +import Box from '@mui/material/Box'; +import Rating from '@mui/material/Rating'; +import { + GridRenderCellParams, + GridRenderEditCellParams, + useGridApiContext, +} from '@mui/x-data-grid'; + +interface RatingValueProps { + value: number; +} + +const RatingValue = React.memo(function RatingValue(props: RatingValueProps) { + const { value } = props; + return ( + + {' '} + {Math.round(Number(value) * 10) / 10} + + ); +}); + +function EditRating(props: GridRenderEditCellParams) { + const { id, value, field } = props; + + const apiRef = useGridApiContext(); + + const changedThroughKeyboard = React.useRef(false); + + const handleChange = async (event: any) => { + await apiRef.current.setEditCellValue( + { id, field, value: Number(event.target.value) }, + event, + ); + if (!changedThroughKeyboard.current) { + apiRef.current.stopCellEditMode({ id, field }); + } + changedThroughKeyboard.current = false; + }; + + const handleRef = (element: HTMLElement | undefined) => { + if (element) { + if (value !== 0) { + element.querySelector(`input[value="${value}"]`)!.focus(); + } else { + element.querySelector('input[value=""]')!.focus(); + } + } + }; + + const handleKeyDown = (event: React.KeyboardEvent) => { + if (event.key.startsWith('Arrow')) { + changedThroughKeyboard.current = true; + } else { + changedThroughKeyboard.current = false; + } + }; + + return ( + + + {Number(value)} + + ); +} + +export function renderRating(params: GridRenderCellParams) { + if (params.value == null) { + return ''; + } + + return ; +} + +export function renderEditRating(params: GridRenderEditCellParams) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/sparkline.js b/docs/data/data-grid/custom-columns/cell-renderers/sparkline.js new file mode 100644 index 0000000000000..b6b083cf7b549 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/sparkline.js @@ -0,0 +1,16 @@ +import * as React from 'react'; +import { SparkLineChart } from '@mui/x-charts/SparkLineChart'; + +export function renderSparkline(params) { + if (params.value == null) { + return ''; + } + + return ( + + ); +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/sparkline.tsx b/docs/data/data-grid/custom-columns/cell-renderers/sparkline.tsx new file mode 100644 index 0000000000000..c5e44fa6c8858 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/sparkline.tsx @@ -0,0 +1,17 @@ +import * as React from 'react'; +import { SparkLineChart } from '@mui/x-charts/SparkLineChart'; +import { GridRenderCellParams } from '@mui/x-data-grid'; + +export function renderSparkline(params: GridRenderCellParams) { + if (params.value == null) { + return ''; + } + + return ( + + ); +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/status.js b/docs/data/data-grid/custom-columns/cell-renderers/status.js new file mode 100644 index 0000000000000..f67ec6746c923 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/status.js @@ -0,0 +1,156 @@ +import * as React from 'react'; +import Chip from '@mui/material/Chip'; +import ListItemIcon from '@mui/material/ListItemIcon'; +import ListItemText from '@mui/material/ListItemText'; +import MenuItem from '@mui/material/MenuItem'; + +import Select from '@mui/material/Select'; +import { styled } from '@mui/material/styles'; +import ReportProblemIcon from '@mui/icons-material/ReportProblem'; +import InfoIcon from '@mui/icons-material/Info'; +import AutorenewIcon from '@mui/icons-material/Autorenew'; +import DoneIcon from '@mui/icons-material/Done'; +import { + GridEditModes, + useGridApiContext, + useGridRootProps, +} from '@mui/x-data-grid'; + +export const STATUS_OPTIONS = ['Open', 'PartiallyFilled', 'Filled', 'Rejected']; + +const StyledChip = styled(Chip)(({ theme }) => ({ + justifyContent: 'left', + '& .icon': { + color: 'inherit', + }, + '&.Open': { + color: (theme.vars || theme).palette.info.dark, + border: `1px solid ${(theme.vars || theme).palette.info.main}`, + }, + '&.Filled': { + color: (theme.vars || theme).palette.success.dark, + border: `1px solid ${(theme.vars || theme).palette.success.main}`, + }, + '&.PartiallyFilled': { + color: (theme.vars || theme).palette.warning.dark, + border: `1px solid ${(theme.vars || theme).palette.warning.main}`, + }, + '&.Rejected': { + color: (theme.vars || theme).palette.error.dark, + border: `1px solid ${(theme.vars || theme).palette.error.main}`, + }, +})); + +const Status = React.memo((props) => { + const { status } = props; + + let icon = null; + if (status === 'Rejected') { + icon = ; + } else if (status === 'Open') { + icon = ; + } else if (status === 'PartiallyFilled') { + icon = ; + } else if (status === 'Filled') { + icon = ; + } + + let label = status; + if (status === 'PartiallyFilled') { + label = 'Partially Filled'; + } + + return ( + + ); +}); + +function EditStatus(props) { + const { id, value, field } = props; + const rootProps = useGridRootProps(); + const apiRef = useGridApiContext(); + + const handleChange = async (event) => { + const isValid = await apiRef.current.setEditCellValue({ + id, + field, + value: event.target.value, + }); + + if (isValid && rootProps.editMode === GridEditModes.Cell) { + apiRef.current.stopCellEditMode({ id, field, cellToFocusAfter: 'below' }); + } + }; + + const handleClose = (event, reason) => { + if (reason === 'backdropClick') { + apiRef.current.stopCellEditMode({ id, field, ignoreModifications: true }); + } + }; + + return ( + + ); +} + +export function renderStatus(params) { + if (params.value == null) { + return ''; + } + + return ; +} + +export function renderEditStatus(params) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/cell-renderers/status.tsx b/docs/data/data-grid/custom-columns/cell-renderers/status.tsx new file mode 100644 index 0000000000000..85bb29b226994 --- /dev/null +++ b/docs/data/data-grid/custom-columns/cell-renderers/status.tsx @@ -0,0 +1,162 @@ +import * as React from 'react'; +import Chip from '@mui/material/Chip'; +import ListItemIcon from '@mui/material/ListItemIcon'; +import ListItemText from '@mui/material/ListItemText'; +import MenuItem from '@mui/material/MenuItem'; +import type { MenuProps } from '@mui/material/Menu'; +import Select, { SelectProps } from '@mui/material/Select'; +import { styled } from '@mui/material/styles'; +import ReportProblemIcon from '@mui/icons-material/ReportProblem'; +import InfoIcon from '@mui/icons-material/Info'; +import AutorenewIcon from '@mui/icons-material/Autorenew'; +import DoneIcon from '@mui/icons-material/Done'; +import { + GridEditModes, + GridRenderCellParams, + GridRenderEditCellParams, + useGridApiContext, + useGridRootProps, +} from '@mui/x-data-grid'; + +export const STATUS_OPTIONS = ['Open', 'PartiallyFilled', 'Filled', 'Rejected']; + +interface StatusProps { + status: string; +} + +const StyledChip = styled(Chip)(({ theme }) => ({ + justifyContent: 'left', + '& .icon': { + color: 'inherit', + }, + '&.Open': { + color: (theme.vars || theme).palette.info.dark, + border: `1px solid ${(theme.vars || theme).palette.info.main}`, + }, + '&.Filled': { + color: (theme.vars || theme).palette.success.dark, + border: `1px solid ${(theme.vars || theme).palette.success.main}`, + }, + '&.PartiallyFilled': { + color: (theme.vars || theme).palette.warning.dark, + border: `1px solid ${(theme.vars || theme).palette.warning.main}`, + }, + '&.Rejected': { + color: (theme.vars || theme).palette.error.dark, + border: `1px solid ${(theme.vars || theme).palette.error.main}`, + }, +})); + +const Status = React.memo((props: StatusProps) => { + const { status } = props; + + let icon: any = null; + if (status === 'Rejected') { + icon = ; + } else if (status === 'Open') { + icon = ; + } else if (status === 'PartiallyFilled') { + icon = ; + } else if (status === 'Filled') { + icon = ; + } + + let label: string = status; + if (status === 'PartiallyFilled') { + label = 'Partially Filled'; + } + + return ( + + ); +}); + +function EditStatus(props: GridRenderEditCellParams) { + const { id, value, field } = props; + const rootProps = useGridRootProps(); + const apiRef = useGridApiContext(); + + const handleChange: SelectProps['onChange'] = async (event) => { + const isValid = await apiRef.current.setEditCellValue({ + id, + field, + value: event.target.value, + }); + + if (isValid && rootProps.editMode === GridEditModes.Cell) { + apiRef.current.stopCellEditMode({ id, field, cellToFocusAfter: 'below' }); + } + }; + + const handleClose: MenuProps['onClose'] = (event, reason) => { + if (reason === 'backdropClick') { + apiRef.current.stopCellEditMode({ id, field, ignoreModifications: true }); + } + }; + + return ( + + ); +} + +export function renderStatus(params: GridRenderCellParams) { + if (params.value == null) { + return ''; + } + + return ; +} + +export function renderEditStatus(params: GridRenderEditCellParams) { + return ; +} diff --git a/docs/data/data-grid/custom-columns/custom-columns.md b/docs/data/data-grid/custom-columns/custom-columns.md index 118a6905b663a..ebabc45e057fb 100644 --- a/docs/data/data-grid/custom-columns/custom-columns.md +++ b/docs/data/data-grid/custom-columns/custom-columns.md @@ -44,6 +44,18 @@ You can change the date format by importing different locale (`en-US` locale is See [Localization](/x/react-date-pickers/localization/) for more information. ::: +## Full example + +The demo below shows the most common custom column renderers used across our demos. + +:::success +You can copy the column definitions and custom cell renderers from the demo source code. + +All column definitions are located in the main component file, while each cell renderer is in a separate file. +::: + +{{"demo": "CustomColumnFullExample.js", "bg": "inline"}} + ## API - [DataGrid](/x/api/data-grid/data-grid/) diff --git a/docs/data/data-grid/faq/faq.md b/docs/data/data-grid/faq/faq.md index 7c3d17e7c94b9..2aec0da7e1744 100644 --- a/docs/data/data-grid/faq/faq.md +++ b/docs/data/data-grid/faq/faq.md @@ -124,7 +124,14 @@ It only impacts the rendering part and does not impact the internal calculations ## What is the purpose of useDemoData hook used in the documentation examples? The `useDemoData` hook is a utility hook from the `@mui/x-data-grid-generator` package. -It generates random data for the Data Grid. It is often used in documentation examples to provide realistic data without polluting the code with data generation logic. +It contains columns definitions and generates random data for the Data Grid. +It is often used in our demos to provide realistic data without polluting the code with data generation logic. + +:::success +Looking for the column definitions and custom cell renderers from the `useDemoData` hook? + +Check out the [Custom columns demo](/x/react-data-grid/custom-columns/#full-example) where you can copy them from the demo source code. +::: Here's how it's used: diff --git a/packages/x-data-grid-generator/src/columns/commodities.columns.tsx b/packages/x-data-grid-generator/src/columns/commodities.columns.tsx index 60c9f269b87f1..56f6f1e9528a7 100644 --- a/packages/x-data-grid-generator/src/columns/commodities.columns.tsx +++ b/packages/x-data-grid-generator/src/columns/commodities.columns.tsx @@ -174,6 +174,7 @@ export const getCommodityColumns = (editable = false): GridColDefGenerator[] => }, { field: 'incoTerm', + headerName: 'Incoterm', generateData: randomIncoterm, renderCell: renderIncoterm, renderEditCell: renderEditIncoterm, diff --git a/packages/x-data-grid-generator/src/renderer/renderEditCountry.tsx b/packages/x-data-grid-generator/src/renderer/renderEditCountry.tsx index f963d73b6563f..52157b6a69ea8 100644 --- a/packages/x-data-grid-generator/src/renderer/renderEditCountry.tsx +++ b/packages/x-data-grid-generator/src/renderer/renderEditCountry.tsx @@ -54,6 +54,7 @@ function EditCountry(props: GridRenderEditCellParams) { }, }} {...optionProps} + key={option.code} >