-
Notifications
You must be signed in to change notification settings - Fork 1.2k
/
Copy pathuseTextField.ts
114 lines (104 loc) · 3.99 KB
/
useTextField.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
/*
* Copyright 2020 Adobe. All rights reserved.
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. You may obtain a copy
* of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
* OF ANY KIND, either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/
import {AriaTextFieldProps} from '@react-types/textfield';
import {ChangeEvent, InputHTMLAttributes, LabelHTMLAttributes, RefObject, TextareaHTMLAttributes} from 'react';
import {ElementType} from 'react';
import {filterDOMProps, mergeProps} from '@react-aria/utils';
import {useFocusable} from '@react-aria/focus';
import {useKeyboard} from '@react-aria/interactions';
import {useLabel} from '@react-aria/label';
interface TextFieldAria {
/** Props for the input element. */
inputProps: InputHTMLAttributes<HTMLInputElement> | TextareaHTMLAttributes<HTMLTextAreaElement>,
/** Props for the text field's visible label element (if any). */
labelProps: LabelHTMLAttributes<HTMLLabelElement>
}
interface AriaTextFieldOptions extends AriaTextFieldProps {
/**
* The HTML element used to render the input, e.g. 'input', or 'textarea'.
* It determines whether certain HTML attributes will be included in `inputProps`.
* For example, [`type`](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input#attr-type).
* @default 'input'
*/
inputElementType?: ElementType
}
/**
* Provides the behavior and accessibility implementation for a text field.
* @param props - Props for the text field.
* @param ref - Ref to the HTML input or textarea element.
*/
export function useTextField(
props: AriaTextFieldOptions,
ref: RefObject<HTMLInputElement | HTMLTextAreaElement>
): TextFieldAria {
let {
inputElementType = 'input',
isDisabled = false,
isRequired = false,
isReadOnly = false,
validationState,
type = 'text',
onChange = () => {},
onKeyDown = () => {},
onKeyUp = () => {}
} = props;
let {focusableProps} = useFocusable(props, ref);
let {labelProps, fieldProps} = useLabel(props);
let domProps = filterDOMProps(props, {labelable: true});
let {keyboardProps} = useKeyboard({...props, onKeyDown, onKeyUp});
const inputOnlyProps = {
type,
pattern: props.pattern
};
return {
labelProps,
inputProps: mergeProps(
domProps,
inputElementType === 'input' && inputOnlyProps,
{
disabled: isDisabled,
readOnly: isReadOnly,
'aria-required': isRequired || undefined,
'aria-invalid': validationState === 'invalid' || undefined,
'aria-errormessage': props['aria-errormessage'],
'aria-activedescendant': props['aria-activedescendant'],
'aria-autocomplete': props['aria-autocomplete'],
'aria-haspopup': props['aria-haspopup'],
value: props.value,
defaultValue: props.value ? undefined : props.defaultValue,
onChange: (e: ChangeEvent<HTMLInputElement>) => onChange(e.target.value),
autoComplete: props.autoComplete,
maxLength: props.maxLength,
minLength: props.minLength,
name: props.name,
placeholder: props.placeholder,
inputMode: props.inputMode,
// Clipboard events
onCopy: props.onCopy,
onCut: props.onCut,
onPaste: props.onPaste,
// Composition events
onCompositionEnd: props.onCompositionEnd,
onCompositionStart: props.onCompositionStart,
onCompositionUpdate: props.onCompositionUpdate,
// Selection events
onSelect: props.onSelect,
// Input events
onBeforeInput: props.onBeforeInput,
onInput: props.onInput,
...focusableProps,
...fieldProps,
...keyboardProps
}
)
};
}