Skip to content

Commit df533c3

Browse files
committed
feat: add native input DOM to follow standard behavior of HTML
1 parent c93ff10 commit df533c3

23 files changed

+457
-187
lines changed

README.md

+1
Original file line numberDiff line numberDiff line change
@@ -132,6 +132,7 @@ export default () => (
132132
| optionRender | Custom rendering options | (oriOption: FlattenOptionData\<BaseOptionType\> , info: { index: number }) => React.ReactNode | - |
133133
| labelRender | Custom rendering label | (props: LabelInValueType) => React.ReactNode | - |
134134
| maxCount | The max number of items can be selected | number | - |
135+
| nativeInputProps | Passing props to the native input | `React.InputHTMLAttributes<HTMLInputElement>` | - |
135136

136137
### Methods
137138

src/BaseSelect.tsx

+3
Original file line numberDiff line numberDiff line change
@@ -133,6 +133,7 @@ export interface BaseSelectProps extends BaseSelectPrivateProps, React.AriaAttri
133133
tagRender?: (props: CustomTagProps) => React.ReactElement;
134134
direction?: 'ltr' | 'rtl';
135135
maxLength?: number;
136+
nativeInputProps?: React.InputHTMLAttributes<HTMLInputElement>;
136137

137138
// MISC
138139
tabIndex?: number;
@@ -219,6 +220,7 @@ const BaseSelect = React.forwardRef<BaseSelectRef, BaseSelectProps>((props, ref)
219220
tagRender,
220221
direction,
221222
omitDomProps,
223+
nativeInputProps,
222224

223225
// Value
224226
displayValues,
@@ -792,6 +794,7 @@ const BaseSelect = React.forwardRef<BaseSelectRef, BaseSelectProps>((props, ref)
792794
onSearchSubmit={onInternalSearchSubmit}
793795
onRemove={onSelectorRemove}
794796
tokenWithEnter={tokenWithEnter}
797+
nativeInputProps={nativeInputProps}
795798
/>
796799
)}
797800
</SelectTrigger>

src/SelectNativeInput.tsx

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
import classNames from 'classnames';
2+
import React from 'react';
3+
4+
interface SelectNativeInputProps extends React.InputHTMLAttributes<HTMLInputElement> {
5+
prefixCls?: string;
6+
}
7+
8+
export default React.forwardRef<HTMLInputElement, SelectNativeInputProps>(
9+
function SelectNativeInput(props, ref) {
10+
const { prefixCls, className, ...rest } = props;
11+
12+
return (
13+
<input
14+
aria-hidden
15+
tabIndex={-1}
16+
type="hidden"
17+
readOnly
18+
ref={ref}
19+
className={classNames(`${prefixCls}-native-input`, className)}
20+
{...rest}
21+
/>
22+
);
23+
},
24+
);

src/Selector/MultipleSelector.tsx

+12
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import Input from './Input';
99
import useLayoutEffect from '../hooks/useLayoutEffect';
1010
import type { DisplayValueType, RenderNode, CustomTagProps, RawValueType } from '../BaseSelect';
1111
import { getTitle } from '../utils/commonUtil';
12+
import SelectNativeInput from '../SelectNativeInput';
1213

1314
function itemKey(value: DisplayValueType) {
1415
return value.key ?? value.value;
@@ -42,6 +43,7 @@ const SelectSelector: React.FC<SelectorProps> = (props) => {
4243
const {
4344
id,
4445
prefixCls,
46+
nativeInputProps,
4547

4648
values,
4749
open,
@@ -232,9 +234,19 @@ const SelectSelector: React.FC<SelectorProps> = (props) => {
232234
/>
233235
);
234236

237+
const selectNativeInputValue = values
238+
.filter((item) => item.value !== null || item.value !== undefined)
239+
.map((item) => String(item.value))
240+
.join(',');
241+
235242
return (
236243
<>
237244
{selectionNode}
245+
<SelectNativeInput
246+
value={selectNativeInputValue}
247+
prefixCls={prefixCls}
248+
{...nativeInputProps}
249+
/>
238250
{!values.length && !inputValue && (
239251
<span className={`${selectionPrefixCls}-placeholder`}>{placeholder}</span>
240252
)}

src/Selector/SingleSelector.tsx

+8
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import pickAttrs from 'rc-util/lib/pickAttrs';
33
import Input from './Input';
44
import type { InnerSelectorProps } from '.';
55
import { getTitle } from '../utils/commonUtil';
6+
import SelectNativeInput from '../SelectNativeInput';
67

78
interface SelectorProps extends InnerSelectorProps {
89
inputElement: React.ReactElement;
@@ -24,6 +25,7 @@ const SingleSelector: React.FC<SelectorProps> = (props) => {
2425
values,
2526
placeholder,
2627
tabIndex,
28+
nativeInputProps,
2729

2830
showSearch,
2931
searchValue,
@@ -106,6 +108,12 @@ const SingleSelector: React.FC<SelectorProps> = (props) => {
106108
/>
107109
</span>
108110

111+
<SelectNativeInput
112+
value={item?.value}
113+
prefixCls={prefixCls}
114+
{...nativeInputProps}
115+
/>
116+
109117
{/* Display value */}
110118
{!combobox && item ? (
111119
<span

src/Selector/index.tsx

+3-1
Original file line numberDiff line numberDiff line change
@@ -15,14 +15,15 @@ import type { ScrollTo } from 'rc-virtual-list/lib/List';
1515
import MultipleSelector from './MultipleSelector';
1616
import SingleSelector from './SingleSelector';
1717
import useLock from '../hooks/useLock';
18-
import type { CustomTagProps, DisplayValueType, Mode, RenderNode } from '../BaseSelect';
18+
import type { BaseSelectProps, CustomTagProps, DisplayValueType, Mode, RenderNode } from '../BaseSelect';
1919
import { isValidateOpenKey } from '../utils/keyUtil';
2020

2121
export interface InnerSelectorProps {
2222
prefixCls: string;
2323
id: string;
2424
mode: Mode;
2525
title?: string;
26+
nativeInputProps?: BaseSelectProps['nativeInputProps']
2627

2728
inputRef: React.Ref<HTMLInputElement | HTMLTextAreaElement>;
2829
placeholder?: React.ReactNode;
@@ -65,6 +66,7 @@ export interface SelectorProps {
6566
autoClearSearchValue: boolean;
6667
inputElement: JSX.Element;
6768
maxLength?: number;
69+
nativeInputProps?: BaseSelectProps['nativeInputProps']
6870

6971
autoFocus?: boolean;
7072
activeDescendantId?: string;

tests/Accessibility.test.tsx

+5-5
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ describe('Select.Accessibility', () => {
1818
it('pass aria info to internal input', () => {
1919
const MySelect = Select as any;
2020
const wrapper = mount(<MySelect aria-label="light" data-attr="bamboo" useless="2333" />);
21-
expect(wrapper.find('input').props()).toEqual(
21+
expect(wrapper.find('.rc-select-selection-search-input').props()).toEqual(
2222
expect.objectContaining({
2323
'aria-label': 'light',
2424
}),
@@ -48,20 +48,20 @@ describe('Select.Accessibility', () => {
4848
);
4949

5050
// First Match
51-
wrapper.find('input').simulate('change', { target: { value: 'b' } });
51+
wrapper.find('.rc-select-selection-search-input').simulate('change', { target: { value: 'b' } });
5252
jest.runAllTimers();
5353

5454
expectOpen(wrapper);
5555
expect(
5656
wrapper.find('.rc-select-item-option-active .rc-select-item-option-content').text(),
5757
).toEqual('Bamboo');
5858

59-
wrapper.find('input').simulate('keyDown', { which: KeyCode.ENTER });
59+
wrapper.find('.rc-select-selection-search-input').simulate('keyDown', { which: KeyCode.ENTER });
6060
expectOpen(wrapper, false);
6161

6262
// Next Match
63-
wrapper.find('input').simulate('change', { target: { value: '' } });
64-
wrapper.find('input').simulate('change', { target: { value: 'g' } });
63+
wrapper.find('.rc-select-selection-search-input').simulate('change', { target: { value: '' } });
64+
wrapper.find('.rc-select-selection-search-input').simulate('change', { target: { value: 'g' } });
6565
jest.runAllTimers();
6666

6767
expectOpen(wrapper);

0 commit comments

Comments
 (0)