# TextInput
A control for entering text.
## Import
```tsx
import { TextInput } from '@coinbase/cds-web/controls/TextInput'
```
## Examples
**Note** TextField extends props from [HTMLInputElement](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input#attributes) on web. On mobile, it extends [TextInputProps](https://reactnative.dev/docs/textinput#props) from react-native.
#### Input Label
Default composition of Inputs.
```jsx live
COPY
}
/>
Use the compact variant when space is tight.
```
#### Accessible Text Inputs
TextInput comes with an accessibilityLabel prop. If no accessibilityLabel is passed, it will use the label as the accessibilityLabel. If you want an accessibilityLabel that differs from the Label, you can set this prop.
Here, since no accessibilityLabel is passed, the accessibilityLabel will be "Email".
```jsx
```
Example of passing an accessibilityLabel. For web, this will set aria-label="Enter a Coinbase Email" under the hood
```jsx
```
:::tip Accessibility tip
Like any component system, much of the responsibility for building accessible UIs is in your hands as the consumer to properly implement the component composition. We'll do our best to provide sane fallbacks, but here are the biggest gotchas for `TextInput`s you can watch out for.
##### `aria-*` attr overrides
Any time you use `variant='negative'`, we assume you're showing an error state. If for some reason this is _not_ the case, you will want to use `aria-invalid={false}` to override the default configuration.
##### Message format
It's also advised you always format `helperText` with `Error: ${errorMessage}`. We'd do that for you, but _i18n_ isn't baked into CDS. Take a look at the example below:
:::
```jsx live
```
#### Placeholder Text
```jsx live
```
#### Borderless TextInput (web)
A borderless TextInput SHOULD NOT be used alone. It should be used
with a TypeAhead component.
```jsx live
```
#### Helper Text
##### Default Sentiment
```jsx live
Default sentimentPositive sentiment}
/>
Negative Sentiment}
/>
```
#### Color Surge Enabled
```jsx live
```
#### Content Alignment
```jsx live
Left aligned (default): Right aligned (with compact):
```
#### Label Variants
TextInput supports two label variants: `outside` (default) and `inside`. Note that the `compact` prop, when set to true, will override label variant preference.
:::warning
When using the `inside` label variant, you should always include a `placeholder` prop.
:::
```jsx live
Outside label (default):Inside label:Inside label (with start content):}
placeholder="Search for anything"
/>
Inside label (with end content):}
placeholder="Enter your password"
/>
```
#### StartContent & EndContent
##### Examples of Input Objects placed at the Start
```jsx live
function StartContentExamples() {
return (
Asset: Asset objects are not interactive
}
placeholder="HaeJiWplJohn6W42eCq0Qqft0"
/>
Icon: Icon objects are not interactive.
} placeholder="1234" />
IconButton: The most common use case for Icon Button at the start of a
Text Field is search.
}
placeholder="Search for anything"
/>
);
}
```
#### Read Only
TextInput supports a read-only state which is visually distinct from the disabled state. Read-only inputs have a secondary background color and can still be focused.
```jsx live
}
/>
```
### Example of Input Objects placed at the End
Here are some examples and best practices when using end content in a TextField.
```jsx live
Icon: Icon objects are not interactive.
}
/>
The most common use case for placing a text object at the end of an input is currency. This
object is not interactive.
You can add a Text Button object at the end of an Input. "Copy" is a great example of this.
COPY
}
/>
```
#### Password input
Password Input - Use Icon Buttons at the end for actions like showing a password or clearing text from an input.
> a11y tip: Always provide an `accessibilityLabel` to start/end nodes to clearly communicate state/actions
```jsx live
function PasswordInput() {
const [isVisible, setIsVisible] = useState(false);
const type = useMemo(() => (isVisible ? 'text' : 'password'), [isVisible]);
return (
setIsVisible((isVisible) => !setIsVisible)}
accessibilityLabel={isVisible ? 'Hide password' : 'Show password'}
/>
}
/>
);
}
```
#### Link + Icon Button
If needed, you can add a Link + Icon Button like this example here. Use this sparingly and only at the End of an Input.
```jsx live
function CopyTextField() {
const [copied, setCopied] = useState(false);
const [variant, setVariant] = useState('foregroundMuted');
const [helperText, setHelperText] = useState('');
useEffect(() => {
if (copied) {
setVariant('positive');
setHelperText('Your token has been copied!');
} else {
setVariant('foregroundMuted');
setHelperText('');
}
}, [copied]);
const handleOnChange = useCallback(() => {
setVariant('foregroundMuted');
setCopied(false);
setHelperText('');
}, []);
return (
setCopied(true)} variant="caption" color={variant}>
{copied ? 'copied' : 'copy'}
}
onChange={handleOnChange}
variant={variant}
helperText={helperText}
label="API Access Token"
/>
);
}
```
#### Disabled
```jsx live
```
#### TextArea Example (mobile)
On mobile, TextInput is versatile enough to support
a "TextArea" as well. You just need to add multiline prop.
Here is an example
```jsx
const [text, onChangeText] = useState('');
;
```
#### Example of a Form
We recommend that you use spacing 3 when building stacked forms.
```jsx live
function FormExample() {
const gap = 3;
const onSubmit = useCallback((e) => {
e.preventDefault();
console.log(e.currentTarget.nodeValue);
alert('Submitted');
}, []);
return (
);
}
```
#### Example of a Sign Up Form
```jsx live
```
### Testing
#### Testing different parts of the input
You can also use the testIDMap to test different parts
of the TextInput. If you use testID, it will add the testID to the root
of the TextInput.
```jsx live
function testExample() {
const testIDMap = useMemo(() => {
return {
input: 'input-id',
helperText: 'helperText-id',
label: 'label-id',
start: 'start-id',
end: 'end-id',
};
}, []);
return (
}
end={}
/>
);
}
```
### Date Picker Example
You can construct a DatePicker using TextInput
```jsx live
function DatePicker() {
return ;
}
```
### TextInput While Keyboard Is Open (mobile)
If you have the keyboard open, then closing the keyboard and interacting with the text input requires 2 taps, which isn't a great user experience.
To fix this issue, you can wrap the TextInput in a ScrollView, and set keyboardShouldPersistTaps="always".
```jsx
function TextInputKeyboardExample() {
return (
);
}
```
## Props
| Prop | Type | Required | Default | Description |
| --- | --- | --- | --- | --- |
| `align` | `center \| start \| end \| justify` | No | `start` | Aligns text inside input and helperText |
| `borderRadius` | `0 \| 100 \| 200 \| 300 \| 400 \| 500 \| 600 \| 700 \| 800 \| 900 \| 1000` | No | `200` | Leverage one of the borderRadius styles we offer to round the corners of the input. |
| `bordered` | `boolean` | No | `true` | Adds border to input |
| `compact` | `boolean` | No | `false` | Enables compact variation |
| `disabled` | `boolean` | No | `false` | Toggles input interactability and opacity |
| `enableColorSurge` | `boolean` | No | `-` | Enable Color Surge motion |
| `end` | `null \| string \| number \| false \| true \| ReactElement> \| Iterable \| ReactPortal` | No | `-` | Adds content to the end of the inner input. Refer to diagram for location of endNode in InputStack component |
| `height` | `((Height \| { base?: Height; phone?: Height \| undefined; tablet?: Height \| undefined; desktop?: Height \| undefined; }) & (string \| number)) \| undefined` | No | `-` | Height of input |
| `helperText` | `null \| string \| number \| false \| true \| ReactElement> \| Iterable \| ReactPortal` | No | `-` | For cases where label is not enough information to describe what the text input is for. Can also be used for showing positive/negative messages |
| `helperTextErrorIconAccessibilityLabel` | `string` | No | `'error'` | Accessibility label for helper text error icon when variant=negative |
| `inputNode` | `ReactElement` | No | `-` | Customize the element which the input area will be rendered as. Adds ability to render the input area as a , etc... By default, the input area will be rendered as an . |
| `key` | `Key \| null` | No | `-` | - |
| `label` | `string` | No | `-` | Short messageArea indicating purpose of input |
| `labelVariant` | `inside \| outside` | No | `'outside'` | The variant of the label. Only used when compact is not true. |
| `onChange` | `ChangeEventHandler` | No | `-` | - |
| `onClick` | `(MouseEventHandler & MouseEventHandler)` | No | `-` | Callback fired when pressed/clicked |
| `placeholder` | `string` | No | `-` | Placeholder text displayed inside of the input. Will be replaced if there is a value. |
| `ref` | `((instance: HTMLInputElement \| null) => void) \| RefObject \| null` | No | `-` | - |
| `start` | `null \| string \| number \| false \| true \| ReactElement> \| Iterable \| ReactPortal` | No | `-` | Adds content to the start of the inner input. Refer to diagram for location of startNode in InputStack component |
| `suffix` | `string` | No | `-` | Adds suffix text to the end of input |
| `testID` | `string` | No | `-` | Used to locate this element in unit and end-to-end tests. Under the hood, testID translates to data-testid on Web. On Mobile, testID stays the same - testID |
| `testIDMap` | `{ start?: string; end?: string \| undefined; label?: string \| undefined; helperText?: string \| undefined; } \| undefined` | No | `-` | Add ability to test individual parts of the input |
| `type` | `number \| color \| button \| search \| time \| image \| text \| hidden \| string & {} \| email \| checkbox \| radio \| tel \| url \| date \| submit \| reset \| datetime-local \| file \| month \| password \| range \| week` | No | `-` | - |
| `value` | `string \| number \| readonly string[]` | No | `-` | - |
| `variant` | `primary \| secondary \| positive \| negative \| foregroundMuted \| foreground` | No | `foregroundMuted` | Determines the sentiment of the input. Because we allow startContent and endContent to be custom ReactNode, the content placed inside these slots will not change colors according to the variant. You will have to add that yourself |
| `width` | `ResponsiveProp>` | No | `100%` | Width of input as a percentage string. |