layout | title |
---|---|
default |
SX |
All react-admin components expose an sx
prop, which allows to customize the component style. This sx
prop is kind of like React's style
prop, but it's more powerful. It uses the CSS-in-JS solution offered by MUI, MUI System, powered by emotion.
The sx
prop accepts an object with CSS properties as keys and values, just like the style
prop. It uses the JavaScript name of CSS properties (e.g. fontWeight
instead of font-weight
).
{% raw %}
import { NumberField, List, Datagrid, TextField, EditButton } from 'react-admin';
export const ProductList = () => (
<List>
<Datagrid>
<TextField source="sku" />
<TextField source="price" />
<EditButton sx={{ fontWeight: "bold" }}/>
</Datagrid>
</List>
);
{% endraw %}
In the rendered HTML, this will result in a custom class name, like EditButton-root-123
. This class name will be added to the DOM element, and the corresponding CSS will be injected in the <head>
of the document.
The sx
prop supports CSS syntax including child and pseudo-selectors, media queries, raw CSS values, and more.
Here is another example, taken from the CRM demo:
{% raw %}
export const Welcome = () => (
<Card
sx={{
background: `#c5dedd`,
color: 'rgba(0, 0, 0, 0.87)',
p: 1,
mb: 1,
mt: 2,
[`& .MuiCardActions-root`]: {
p: 2,
mt: -2,
mb: -1,
flexDirection: 'column',
'& a': {
mb: 1,
color: 'rgba(0, 0, 0, 0.87)',
backgroundColor: 'white',
marginLeft: '0 !important',
},
},
}}
>
// ...
</Card>
);
{% endraw %}
You can use any CSS property (see the list of all accepted CSS properties in the MUI documentation). Take advantage of IDE autocompletion to discover the available properties.
The sx
prop accepts shorthand properties, like m
for margin
or mt
for marginTop
:
{% raw %}
<Box
sx={{
m: 1, // margin: theme.spacing(1)
px: 1, // padding-left, padding-right: theme.spacing(1)
}}
>
{% endraw %}
Common shorthands are:
m
:margin
mt
:margin-top
mr
:margin-right
mb
:margin-bottom
ml
:margin-left
mx
:margin-left, margin-right
my
:margin-top, margin-bottom
p
:padding
pt
:padding-top
pr
:padding-right
pb
:padding-bottom
pl
:padding-left
px
:padding-left, padding-right
py
:padding-top, padding-bottom
bgcolor
:backgroundColor
Some CSS properties accept theme-aware values, i.e. the sx to CSS conversion uses the application theme. Check the material-ui default theme for the list of values you can use.
The typography
property can receive a string, which represents the path in theme.typography
:
{% raw %}
<Box sx={{ typography: 'subtitle2' }} /> // theme.typography.subtitle2
<Box sx={{ typography: 'body1' }} />
<Box sx={{ typography: 'body2' }} />
{% endraw %}
The color
and backgroundColor
properties can receive a string, which represents the path in theme.palette
:
{% raw %}
<Box sx={{ color: 'primary.main' }} />
// equivalent to color: theme => theme.palette.primary.main
<Box sx={{ color: 'secondary.main' }} />
<Box sx={{ color: 'error.main' }} />
<Box sx={{ color: 'warning.main' }} />
<Box sx={{ color: 'info.main' }} />
<Box sx={{ color: 'success.main' }} />
<Box sx={{ color: 'text.primary' }} />
<Box sx={{ color: 'text.secondary' }} />
<Box sx={{ color: 'text.disabled' }} />
{% endraw %}
The value of sizing properties width
, height
, minHeight
, maxHeight
, minWidth
, and maxWidth
is converted to a percentage if it is between [0, 1]
. Otherwise, it is directly set on the CSS property:
{% raw %}
<Box sx={{ width: 1/2 }} /> // equivalent to width: '50%'
<Box sx={{ width: 20 }} /> // equivalent to width: '20px'
{% endraw %}
The spacing properties margin
, padding
, and the corresponding longhand properties multiply the values they receive by the theme.spacing
value (the default for the value is 8px):
{% raw %}
<Box sx={{ margin: 2 }} />
// equivalent to margin: theme => theme.spacing(2)
{% endraw %}
The fontFamily
, fontSize
, fontStyle
, fontWeight
properties map their value to the theme.typography
value:
{% raw %}
<Box sx={{ fontWeight: 'fontWeightLight' }} />
// equivalent to fontWeight: theme.typography.fontWeightLight
{% endraw %}
The sx
prop simplifies the process of defining and implementing responsive breakpoints. You can define a set of breakpoints as an object, using the breakpoint values as keys.
{% raw %}
<Box
sx={{
width: {
xs: 100, // theme.breakpoints.up('xs')
sm: 200, // theme.breakpoints.up('sm')
md: 300, // theme.breakpoints.up('md')
lg: 400, // theme.breakpoints.up('lg')
xl: 500, // theme.breakpoints.up('xl')
},
}}
>
This box has a responsive width.
</Box>
{% endraw %}
Note that each property for a given breakpoint also applies to all larger breakpoints in the set. For example, width: { lg: 100 }
is equivalent to theme.breakpoints.up('lg')
.
You can use this syntax to show or hide a component based on the screen size:
- Hidden on all: {% raw %}
sx={{ display: 'none' }}
{% endraw %} - Hidden only on xs: {% raw %}
sx={{ display: { xs: 'none', sm: 'block' } }}
{% endraw %} - Hidden only on sm: {% raw %}
sx={{ display: { xs: 'block', sm: 'none', md: 'block' } }}
{% endraw %} - Hidden only on md: {% raw %}
sx={{ display: { xs: 'block', md: 'none', lg: 'block' } }}
{% endraw %} - Hidden only on lg: {% raw %}
sx={{ display: { xs: 'block', lg: 'none', xl: 'block' } }}
{% endraw %} - Hidden only on xl: {% raw %}
sx={{ display: { xs: 'block', xl: 'none' } }}
{% endraw %} - Visible only on xs: {% raw %}
sx={{ display: { xs: 'block', sm: 'none' } }}
{% endraw %} - Visible only on sm: {% raw %}
sx={{ display: { xs: 'none', sm: 'block', md: 'none' } }}
{% endraw %} - Visible only on md: {% raw %}
sx={{ display: { xs: 'none', md: 'block', lg: 'none' } }}
{% endraw %} - Visible only on lg: {% raw %}
sx={{ display: { xs: 'none', lg: 'block', xl: 'none' } }}
{% endraw %} - Visible only on xl: {% raw %}
sx={{ display: { xs: 'none', xl: 'block' } }}
{% endraw %}
You can use media queries in the sx
prop:
{% raw %}
<Box
sx={{
// some styles
'@media print': {
width: 300,
},
}}
>
{% endraw %}
Tip: Don't use media queries for responsive values. Use the responsive values syntax instead.
Tip: The displayPrint
property allows you to specify a CSS display value that will only be applied when printing:
{% raw %}
<Box sx={{ displayPrint: 'none' }} /> // equivalent to '@media print': { display: 'none' }
{% endraw %}
You can use pseudo-classes to style a component only when it's in a specific state. For instance, you can use the :hover
pseudo-class to style a component when the mouse is over it:
{% raw %}
<Box
sx={{
// some styles
":hover": {
boxShadow: 6,
},
}}
>
{% endraw %}
Common pseudo-classes used for styling are :hover
, :disabled
, :active
, :focus
, :first-child
, :last-child
. Check the list of CSS pseudo classes for more information.
The real power of the sx
prop is that it allows to override the style of sub-components. For instance, you can use it to style the <Avatar>
inside another element:
{% raw %}
const CustomUserCard = ({ user }) => (
<UserCard
user={user}
sx={{
display: 'flex',
alignItems: 'center',
'& .MuiAvatar-root': {
width: 48,
height: 48,
mr: 1,
},
}}
/>
);
{% endraw %}
Here is another example, illustrating the ability to customize the header of a <Datagrid>
:
{% raw %}
import {
BooleanField,
Datagrid,
DateField,
EditButton,
List,
NumberField,
TextField,
ShowButton,
} from 'react-admin';
import Icon from '@mui/icons-material/Person';
export const VisitorIcon = Icon;
export const PostList = () => (
<List>
<Datagrid
sx={{
backgroundColor: "Lavender",
"& .RaDatagrid-headerCell": {
backgroundColor: "MistyRose",
},
}}
>
<TextField source="id" />
<TextField source="title" />
<DateField source="published_at" sortByOrder="DESC" />
<BooleanField source="commentable" sortable={false} />
<NumberField source="views" sortByOrder="DESC" />
<EditButton />
<ShowButton />
</Datagrid>
</List>
);
{% endraw %}
This example results in:
To guess the name of the subclass to use (like .RaDatagrid-headerCell
above) for customizing a component, you can use the developer tools of your browser:
The react-admin documentation for individual components also lists the classes available for styling. For instance, here is the Datagrid CSS documentation:
Each property in the sx prop can receive a function callback as a value. This is useful when you want to use the theme for calculating a value:
{% raw %}
<Box sx={{ height: (theme) => theme.spacing(10) }} />
{% endraw %}
The sx
prop can also receive a callback when you need to get theme values that are objects:
{% raw %}
<Box
sx={(theme) => ({
...theme.typography.body,
color: theme.palette.primary.main,
})}
/>
{% endraw %}
Tip: If you use TypeScript, you'll have to specify the type of the theme in the callback:
{% raw %}
import { Box, Theme } from '@mui/material';
export const MyComponent = () => (
<Box
sx={{
p: 1,
border: 1,
borderColor: (theme: Theme) => theme.palette.primary.main,
}}
>
Border color with theme value.
</Box>
)
{% endraw %}
Sometimes you want the format to depend on the value. Use useRecordContext
to grab the record in a component, and the sx
prop to apply the format.
The following example shows how to create a new <ColoredNumberField>
component, which renders with red text when its value is less than 0.
{% raw %}
import { useRecordContext, NumberField, List, Datagrid, TextField, EditButton } from 'react-admin';
import type { NumberFieldProps } from 'react-admin';
const ColoredNumberField = (props: NumberFieldProps) => {
const record = useRecordContext();
return (
<NumberField
sx={{ color: record[prop.source] < 0 ? 'red' : 'black' }}
{...props}
/>
);
};
// Ensure the original component defaultProps are still applied as they may be used by its parents (such as the `Show` component):
ColoredNumberField.defaultProps = NumberField.defaultProps;
export const PostList = () => (
<List>
<Datagrid>
<TextField source="id" />
...
<ColoredNumberField source="nb_views" />
<EditButton />
</Datagrid>
</List>
);
{% endraw %}
Tip: if you don't want to create a custom component to apply conditional formatting, you can also use the <WithRecord>
component.
To reuse the same style overrides in different locations across your application, create a reusable component using the Material UI styled()
utility. It's a function that creates a new component based on a source component and custom styles. The basic syntax is styled(Component)(styles) => Component
(where styles
follows the same syntax as the sx
prop).
For instance, to create a custom <Datagrid>
component with the header style defined in the previous section:
// in src/MyDatagrid.js
import { styled } from '@mui/system';
import { Datagrid } from 'react-admin';
export const MyDatagrid = styled(Datagrid)({
backgroundColor: "Lavender",
"& .RaDatagrid-headerCell": {
backgroundColor: "MistyRose",
},
});
You can then use this component instead of react-admin's <Datagrid>
component:
{% raw %}
// in src/post/PostList.js
import {
BooleanField,
- Datagrid,
DateField,
EditButton,
List,
NumberField,
TextField,
ShowButton,
} from 'react-admin';
+import { MyDatagrid } from '../MyDatagrid';
export const PostList = () => (
<List>
- <Datagrid
- sx={{
- backgroundColor: "Lavender",
- "& .RaDatagrid-headerCell": {
- backgroundColor: "MistyRose",
- },
- }}
- >
+ <MyDatagrid>
<TextField source="id" />
<TextField source="title" />
<DateField source="published_at" sortByOrder="DESC" />
<BooleanField source="commentable" sortable={false} />
<NumberField source="views" sortByOrder="DESC" />
<EditButton />
<ShowButton />
+ </MyDatagrid>
- </Datagrid>
</List>
);
{% endraw %}
Again, to guess the name of the subclass to use (like .RaDatagrid-headerCell
above) for customizing a component, you can use the developer tools of your browser, or check the react-admin documentation for individual components (e.g. the Datagrid CSS documentation).
Check The Material UI documentation on the sx
prop for more information.