跳转到内容

Select 选择属性

选择器组件能从一个选项列表中去获得用户所提供的信息。

基础的选择器

我们通常将菜单(Menus)放置在其所点击的元素上,这样的话能够确保当前选定的菜单项显示在点击的元素之上。

<FormControl fullWidth>
  <InputLabel id="demo-simple-select-label">Age</InputLabel>
  <Select
    labelId="demo-simple-select-label"
    id="demo-simple-select"
    value={age}
    label="Age"
    onChange={handleChange}
  >
    <MenuItem value={10}>Ten</MenuItem>
    <MenuItem value={20}>Twenty</MenuItem>
    <MenuItem value={30}>Thirty</MenuItem>
  </Select>
</FormControl>

高级功能

Select 组件的设计原理是和一个原生的 <select> 元素能够互相替代。

若您需要一个更优雅的功能,譬如 combobox,multiselect,autocomplete,async 或者 creatable support,请查看 Autocomplete 组件。 此组件旨在改进 “react-select” 和 “downshift” 这两个包。

属性

选择器组件是通过自定义 InputBase<input> 元素来实现的。 It extends the text field components sub-components, either the OutlinedInput, Input, or FilledInput, depending on the variant selected. 它有着相同的样式和许多相同的属性。 It shares the same styles and many of the same props. 详情请参阅相应组件的 API 文档。

Filled and standard variants

标签和助手文本

With label + helper text

Without label

⚠ Note that when using FormControl with the outlined variant of the Select, you need to provide a label in two places: in the InputLabel component and in the label prop of the Select component (see the above demo).

自动宽度

其他属性

Disabled

Error

Read only

Required

原生选择器

为了提高用户体验,对于在移动设备上使用平台的原生选择器这样的模式,我们是支持的。

<FormControl fullWidth>
  <InputLabel variant="standard" htmlFor="uncontrolled-native">
    Age
  </InputLabel>
  <NativeSelect
    defaultValue={30}
    inputProps={{
      name: 'age',
      id: 'uncontrolled-native',
    }}
  >
    <option value={10}>Ten</option>
    <option value={20}>Twenty</option>
    <option value={30}>Thirty</option>
  </NativeSelect>
</FormControl>

TextField

TextField wrapper 组件是一个完整的表单控件,它包括了标签,输入和帮助文本。 您可以在 在此章节中 查看使用 select 模式的示例。

自定义选择器

你可以参考以下一些例子来自定义组件。 您可以在 重写文档页面 中了解更多有关此内容的信息。

首先,需要设置 InputBase 组件的样式。 一旦设置好了样式,您就可以直接使用文本框组件,也可以将其作为一个 select 的字段提供给 select 组件的 input 属性。 Notice that the "standard" variant is easier to customize, since it does not wrap the contents in a fieldset/legend markup.

🎨 If you are looking for inspiration, you can check MUI Treasury's customization examples.

多重选择

The Select component can handle multiple selections. It's enabled with the multiple prop.

与单项选择一样,您可以通过访问 onChange 的回调函数中的 event.target.value 来提取新的值。 它总是以一个数组的形式出现。

默认值

选中标记

Chip

占位符

原生(Native)

可被控制的打开选择框

You can control the open state of the select with the open prop. Alternatively, it is also possible to set the initial (uncontrolled) open state of the component with the defaultOpen prop.

与对话框组件(Dialog)一起使用

While it's discouraged by the Material Design guidelines, you can use a select inside a dialog.

联动

可以和 ListSubheader 组件一起罗列分类,或者和原生的 <optgroup> 元素一起使用。

无障碍设计

若想正确的给 Select 加上标签,你的 input 控件需要一个额外的带有 label 的 id 属性。 id 的内容需要和 SelectlabelId 值相同,例如:

<InputLabel id="label">年龄</InputLabel>
<Select labelId="label" id="select" value="20">
  <MenuItem value="10">10</MenuItem>
  <MenuItem value="20">20</MenuItem>
</Select>

对于一个 原生选择器,你应该通过将选择元素的 id 属性的值赋给 InputLabelhtmlFor 属性来提及标签。

<TextField id="select" label="Age" value="20" select>
  <MenuItem value="10">Ten</MenuItem>
  <MenuItem value="20">Twenty</MenuItem>
</TextField>

或者,您也可以使用一个带有 idlabelTextField 组件来创建合适的标记和 id:

<InputLabel htmlFor="select">Age</InputLabel>
<NativeSelect id="select">
  <option value="10">Ten</option>
  <option value="20">Twenty</option>
</NativeSelect>

Unstyled

The Select also comes with an unstyled version. It's ideal for doing heavy customizations and minimizing bundle size.

Unstyled component

import SelectUnstyled from '@mui/base/SelectUnstyled';

Basic usage

<CustomSelect defaultValue={10}>
  <StyledOption value={10}>Ten</StyledOption>
  <StyledOption value={20}>Twenty</StyledOption>
  <StyledOption value={30}>Thirty</StyledOption>
</CustomSelect>

The SelectUnstyled is a component that accepts generic props. Due to Typescript limitations, this may cause unexpected behavior when wrapping the component in forwardRef (or other higher-order components). In such cases, the generic argument will be defaulted to unknown and type suggestions will be incomplete. To avoid this, manually cast the resulting component to the correct type (as shown above).

The rest of the demos below will not use forwardRef for brevity.

Controlled select

The SelectUnstyled can be used as either uncontrolled (as shown in the demo above) or controlled component.

Selected value: 10

<CustomSelect value={value} onChange={setValue}>
  <StyledOption value={10}>Ten</StyledOption>
  <StyledOption value={20}>Twenty</StyledOption>
  <StyledOption value={30}>Thirty</StyledOption>
</CustomSelect>

<Paragraph>Selected value: {value}</Paragraph>

Usage with object values

The unstyled select may be used with non-string values.

Selected character:

{
  "name": "Frodo",
  "race": "Hobbit"
}
<CustomSelect value={character} onChange={setCharacter}>
  {characters.map((c) => (
    <StyledOption key={c.name} value={c}>
      {c.name}
    </StyledOption>
  ))}
</CustomSelect>

<Paragraph>Selected character:</Paragraph>
<Pre>{JSON.stringify(character, null, 2)}</Pre>

Customizing the selected value appearance

It is possible to customize the selected value display by providing a function to the renderValue prop. The element returned by this function will be rendered inside the select's button.

<CustomSelect renderValue={renderValue}>
  <StyledOption value={10}>Ten</StyledOption>
  <StyledOption value={20}>Twenty</StyledOption>
  <StyledOption value={30}>Thirty</StyledOption>
</CustomSelect>

Customizing the options' appearance

Options don't have to be plain strings. You can include custom elements to be rendered inside the listbox.

<CustomSelect>
  {countries.map((c) => (
    <StyledOption key={c.code} value={c.code}>
      <img
        loading="lazy"
        width="20"
        src={`https://flagcdn.com/w20/${c.code.toLowerCase()}.png`}
        srcSet={`https://flagcdn.com/w40/${c.code.toLowerCase()}.png 2x`}
        alt={`Flag of ${c.label}`}
      />
      {c.label} ({c.code}) +{c.phone}
    </StyledOption>
  ))}
</CustomSelect>

Grouping

Options can be grouped, similarly to the how the native select element works. Unlike the native select, however, the groups can be nested.

Place the Option components inside OptionGroup to achieve this.

<CustomSelect>
  <CustomOptionGroup label="Hobbits">
    <StyledOption value="Frodo">Frodo</StyledOption>
    <StyledOption value="Sam">Sam</StyledOption>
    <StyledOption value="Merry">Merry</StyledOption>
    <StyledOption value="Pippin">Pippin</StyledOption>
  </CustomOptionGroup>
  <CustomOptionGroup label="Elves">
    <StyledOption value="Galadriel">Galadriel</StyledOption>
    <StyledOption value="Legolas">Legolas</StyledOption>
  </CustomOptionGroup>
</CustomSelect>

Multiselect

To be able to select multiple options at once, use the MultiSelectUnstyled component.

import { MultiSelectUnstyled } from '@mui/base/SelectUnstyled';
<CustomMultiSelect defaultValue={[10, 20]}>
  <StyledOption value={10}>Ten</StyledOption>
  <StyledOption value={20}>Twenty</StyledOption>
  <StyledOption value={30}>Thirty</StyledOption>
  <StyledOption value={40}>Forty</StyledOption>
  <StyledOption value={50}>Fifty</StyledOption>
</CustomMultiSelect>

useSelect hook

import { useSelect } from '@mui/base/SelectUnstyled';

If you need to use Select's functionality in another component, you can use the useSelect hook. It enables maximal customizability at the cost of being low-level.

The following example shows a select that opens when hovered over or focused. It can be controlled by a mouse/touch or a keyboard.

Select a color...
<CustomSelect placeholder="Select a color..." options={options} />