Skip to content

Latest commit

 

History

History
468 lines (382 loc) · 14.3 KB

SX.md

File metadata and controls

468 lines (382 loc) · 14.3 KB
layout title
default
SX

The sx prop

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.

Syntax

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.

Shorthand 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

Theme-Aware Values

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 %}

Responsive Values

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 %}

Media Queries

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 %}

Pseudo-Classes

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.

Styling Children

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:

Visitor List with customized CSS classes

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:

Developer tools

The react-admin documentation for individual components also lists the classes available for styling. For instance, here is the Datagrid CSS documentation:

Datagrid CSS documentation

Callback Values

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 %}

Conditional Formatting

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.

Reusable Components

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).

Going Further

Check The Material UI documentation on the sx prop for more information.