Ícones
Orientação e sugestões para usar ícones com o Material-UI.
Material-UI fornece suporte de ícones de três maneiras:
- Padronizados como ícones do Material Design e exportados como componentes do React (ícones SVG).
- Com o componente SvgIcon, um wrapper React para ícones SVG customizados.
- Com o componente Icon, um wrapper React para ícones de fonte customizados.
Ícones Material
O Material Design padronizou mais de 1.100 ícones oficiais, cada um em cinco "temas" diferentes (veja abaixo). Para cada ícone SVG, exportamos o respectivo componente React do pacote @material-ui/icons. Você pode pesquisar na lista completa destes ícones.
Instalação
Instale o pacote no diretório do seu projeto com:
// usando npm
npm install @material-ui/icons
// usando yarn
yarn add @material-ui/icons
Esses componentes usam o componente SvgIcon
do Material-UI para renderizar o caminho SVG de cada ícone, e por isso tem uma dependência com @materialui/core
.
Se você ainda não estiver usando Material-UI no seu projeto, você pode adicioná-lo com:
// usando npm
npm install @material-ui/core
// usando yarn
yarn add @material-ui/core
Uso
Importe ícones usando uma destas duas opções:
Opção 1:
import AccessAlarmIcon from '@material-ui/icons/AccessAlarm'; import ThreeDRotation from '@material-ui/icons/ThreeDRotation';
Opção 2:
import { AccessAlarm, ThreeDRotation } from '@material-ui/icons';
O mais seguro para o tamanho do pacote é a opção 1, mas alguns desenvolvedores preferem a opção 2. Certifique-se de seguir o guia minimizando o tamanho do pacote antes de usar a segunda abordagem.
Cada ícone Material também tem um "tema": Filled (padrão), Outlined, Rounded, Two-tone, e Sharp. Para importar o componente de ícone com um tema diferente do padrão, acrescente o nome do tema ao nome do ícone. Por exemplo, para usar o ícone @material-ui/icons/Delete
, temos:
- Tema Filled (preenchido que é o padrão) é exportado como
@material-ui/icons/Delete
, - Tema Outlined (contornado) é exportado como
@material-ui/icons/DeleteOutlined
, - Tema Rounded (arredondado) é exportado como
@material-ui/icons/DeleteRounded
, - Tema Two tone (dois tons) é exportado como
@material-ui/icons/DeleteTwoTone
, - Tema Sharp (pontiagudo) é exportado como
@material-ui/icons/DeleteSharp
.
Nota: A especificação Material Design nomeia os ícones usando a nomeação "snake_case" (por exemplo,
delete_forever
,add_a_photo
), enquanto@material-ui/icons
exporta os respectivos ícones usando a nomeação "PascalCase" (por exemploDeleteForever
,AddAPhoto
). Há três exceções a essa regra de nomenclatura:3d_rotation
exportado comoThreeDRotation
,4k
exportado comoFourK
e360
exportado comoThreeSixty
.
Filled
Outlined
Rounded
Two Tone
Sharp
Edge-cases
Testando
Para fins de teste, cada ícone exposto do @material-ui/icons
tem um atributo data-testid
com o nome do ícone. Por exemplo:
import DeleteIcon from '@material-ui/icons/Delete';
tem o seguinte atributo assim que montado:
<svg data-testid="DeleteIcon"></svg>
SvgIcon
Se você precisa de um ícone SVG customizado (não disponível nos ícones Material) você pode usar encapsular com SvgIcon
. Este componente estende o elemento nativo <svg>
:
- Ele vem internamente com a acessibilidade.
- Os elementos SVG devem ser dimensionados para uma visualização de 24x24px, de modo que o ícone resultante possa ser usado como está, ou incluído como filho para outros componentes de Material-UI que usam ícones. (Isso pode ser customizado com o atributo
viewBox
). - Por padrão, o componente herda a cor atual. Opcionalmente, você pode aplicar uma das cores do tema usando a propriedade
color
.
function HomeIcon(props) {
return (
<SvgIcon {...props}>
<path d="M10 20v-6h4v6h5v-8h3L12 3 2 12h3v8z" />
</SvgIcon>
);
}
Cor
<HomeIcon />
<HomeIcon color="primary" />
<HomeIcon color="secondary" />
<HomeIcon color="action" />
<HomeIcon color="disabled" />
<HomeIcon style={{ color: green[500] }} />
<HomeIcon fontSize="small" />
<HomeIcon />
<HomeIcon fontSize="large" />
<HomeIcon style={{ fontSize: 40 }} />
Propriedade Componente
Você pode usar o SvgIcon
para encapsular seus ícones, mesmo que estes estejam salvos no formato .svg
. A biblioteca svgr possui loaders para importar arquivos SVG e usá-los como componentes React. Por exemplo, com webpack:
// webpack.config.js
{
test: /\.svg$/,
use: ['@svgr/webpack'],
}
// ---
import StarIcon from './star.svg';
<SvgIcon component={StarIcon} viewBox="0 0 600 476.6" />
Também é possível usá-lo com "url-loader" ou "file-loader". Esta é a abordagem usada pelo Create React App.
// webpack.config.js
{
test: /\.svg$/,
use: ['@svgr/webpack', 'url-loader'],
}
// ---
import { ReactComponent as StarIcon } from './star.svg';
<SvgIcon component={StarIcon} viewBox="0 0 600 476.6" />
createSvgIcon
O componente utilitário createSvgIcon
é usado para criar ícones Material. Ele pode ser usado para encapsular um caminho SVG com um componente SvgIcon.
const HomeIcon = createSvgIcon(
<path d="M10 20v-6h4v6h5v-8h3L12 3 2 12h3v8z" />,
'Home',
);
<HomeIcon />
<HomeIcon color="primary" />
Fonte Awesome
Se você descobrir que há problemas de leiaute ao usar FontAwesomeIcon de @fortawesome/react-fontawesome
, você pode tentar passar os dados SVG da Font Awesome diretamente para SvgIcon.
Fonte Awesome pode ser usada com o componente Icon
da seguinte forma:
Nota: mdi-material-ui já agrupou cada um desses ícones SVG com o componente SvgIcon
, para que você não precise fazer isso.
FontAwesomeIcon's fullWidth
prop can also be used to approximate the correct dimensions, but it isn't perfect.
Fonte Material icons
MDI
O site materialdesignicons.com fornece mais de 2.000 ícones. For the wanted icon, copy the SVG path
they provide, and use it as the child of the SvgIcon
component, or with createSvgIcon()
.
Nota: A biblioteca mdi-material-ui já agrupou cada um desses ícones SVG com o componente SvgIcon
, para que você não precise fazer isso.
Ícone (ícones de fonte)
O componente Icon
exibirá um ícone de qualquer fonte de ícone que suporte ligadura tipográfica (ligatures). As a prerequisite, you must include one, such as the Material icon font in your project. Para usar um ícone, simplesmente coloque o nome do ícone (font ligature) com o componente Icon
, por exemplo:
import Icon from '@material-ui/core/Icon';
<Icon>star</Icon>
Por padrão, um ícone herdará a cor do texto atual. Opcionalmente, você pode definir a cor do ícone usando uma das propriedades de cor do tema: primary
, secondary
, action
, erro
& disabled
.
Fonte Material icons
Icon
will by default set the correct base class name for the Material Icons font (filled variant). All you need to do is load the font, for instance, via Google Web Fonts:
<link rel="stylesheet" href="https://fonts.googleapis.com/icon?family=Material+Icons" />
<Icon>add_circle</Icon>
<Icon color="primary">add_circle</Icon>
<Icon style={{ color: green[500] }}>add_circle</Icon>
<Icon fontSize="small">add_circle</Icon>
<Icon style={{ fontSize: 30 }}>add_circle</Icon>
Custom font
For other fonts, you can customize the baseline class name using the baseClassName
prop. For instance, you can display two-tone icons with Material Design:
import Icon from '@material-ui/core/Icon';
<link
rel="stylesheet"
href="https://fonts.googleapis.com/css?family=Material+Icons+Two+Tone"
// Import the two tones MD variant ^^^^^^^^
/>;
<Icon baseClassName="material-icons-two-tone">add_circle</Icon>
Global base class name
Modifying the baseClassName
prop for each component usage is repetitive. You can change the default prop globally with the theme
const theme = createMuiTheme({
components: {
MuiIcon: {
defaultProps: {
// Replace the `material-icons` default value.
baseClassName: 'material-icons-two-tone',
},
},
},
});
Then, you can use the two-tone font directly:
<Icon>add_circle</Icon>
Fonte Awesome
Font Awesome can be used with the Icon
component as follows:
<Icon baseClassName="fas" className="fa-plus-circle" />
<Icon baseClassName="fas" className="fa-plus-circle" color="primary" />
<Icon
baseClassName="fas"
className="fa-plus-circle"
style={{ color: green[500] }}
/>
<Icon baseClassName="fas" className="fa-plus-circle" fontSize="small" />
<Icon
baseClassName="fas"
className="fa-plus-circle"
style={{ fontSize: 30 }}
/>
Note that the Font Awesome icons weren't designed like the Material Design icons (compare the two previous demos). The fa icons are cropped to use all the space available. You can adjust for this with a global override:
const theme = createMuiTheme({
components: {
MuiIcon: {
styleOverrides: {
root: {
// Match 24px = 3 * 2 + 1.125 * 16
boxSizing: 'content-box',
padding: 3,
fontSize: '1.125rem',
},
},
},
},
});
<ThemeProvider theme={theme}>
<Chip icon={<MdPhone />} label="Call me" />
<Chip icon={<Icon className="fas fa-phone-alt" />} label="Call me" />
</ThemeProvider>
Fonte vs SVG. Qual abordagem usar?
Both approaches work fine, however there are some subtle differences, especially in terms of performance and rendering quality. Whenever possible SVG is preferred as it allows code splitting, supports more icons, and renders faster and better.
For more details, take a look at why GitHub migrated from font icons to SVG icons.
Acessibilidade
Icons can convey all sorts of meaningful information, so it’s important to ensure they are accessible where appropriate. Há dois casos de uso que você deve considerar:
- Decorative icons that are only being used for visual or branding reinforcement. Se eles forem removidos da página, os usuários ainda entenderiam e poderiam usar sua página.
- Semantic icons are ones that you’re using to convey meaning, rather than just pure decoration. This includes icons without text next to them that are used as interactive controls — buttons, form elements, toggles, etc.
Decorative icons
Se seus ícones são puramente decorativos, você já terminou! O atributo aria-hidden=true
foi adicionado para que seus ícones estejam adequadamente acessíveis (invisíveis).
Semantic icons
Semantic SVG icons
You should include the titleAccess
prop with a meaningful value. The role="img"
attribute and the <title>
element are added so that your icons are correctly accessible.
In the case of focusable interactive elements, for example when used with an icon button, you can use the aria-label
prop:
import IconButton from '@material-ui/core/IconButton';
import SvgIcon from '@material-ui/core/SvgIcon';
// ...
<IconButton aria-label="deletar">
<SvgIcon>
<path d="M20 12l-1.41-1.41L13 16.17V4h-2v12.17l-5.58-5.59L4 12l8 8 8-8z" />
</SvgIcon>
</IconButton>
Semantic font icons
You need to provide a text alternative that is only visible to assistive technologies.
import Icon from '@material-ui/core/Icon';
import Typography from '@material-ui/core/Typography';
// ...
<Icon>add_circle</Icon>
<Typography variant="srOnly">Crie um usuário</Typography>