Components

FormField

The FormField component enhances form inputs with an accessible label and optionally displays additional help text.

Examples #

Import Syntax #

import { FormField } from 'mineral-ui/Form';

Basic Usage #

FormField accepts any input, either to the input prop or children, accessibly connecting the label to it.

<DemoLayout>
  {/* Use the "input" prop for a streamlined approach. */}
  <FormField input={TextInput} label="Name" />

  {/* Or use "children", if you prefer. */}
  <FormField label="Description">
    <TextArea />
  </FormField>
</DemoLayout>

Required #

Marking a FormField as required will display the requiredText and pass along the required prop to the child input. requiredText takes precedence over secondaryText.

<DemoLayout>
  <FormField
    input={TextInput}
    label="Name"
    required />
  <FormField
    input={TextInput}
    label="Nombre"
    required
    requiredText="Necesario" />
</DemoLayout>

Secondary Text #

Use the secondaryText prop to provide brief additional details to the label text. Note that if requiredText is provided, it displays instead of secondaryText.

<FormField
  input={TextInput}
  label="Name"
  secondaryText="Optional" />

Caption #

Use a caption to display not-so-brief additional information, such as help text or, along with a variant, a validation message.

This is help text
<FormField
  input={TextInput}
  label="Name"
  caption="This is help text" />

Variants #

FormField is available in a few variants, mostly for use with validation. Be sure to use the appropriate variant for your intent.

This is help text
This is help text
This is help text
<DemoLayout>
  <FormField
    input={TextInput}
    label="Success"
    caption="This is help text"
    variant="success" />
  <FormField
    input={TextInput}
    label="Warning"
    caption="This is help text"
    variant="warning" />
  <FormField
    input={TextInput}
    label="Danger"
    caption="This is help text"
    variant="danger" />
</DemoLayout>

Visually Hidden Label #

If the purpose of a FormField is obvious from context and adding a label would negatively affect the design, as in the example below, you can use hideLabel to visually hide the label while retaining accessibility.

<DemoLayout>
  <FormField
    input={TextInput}
    label="Address"
    placeholder="1234 Main St" />
  <FormField
    input={TextInput}
    label="Address Line 2"
    placeholder="Apt 101"
    hideLabel />
</DemoLayout>

Validation #

This example demonstrates simple required field validation. An empty input will fail validation while a non-empty input will pass.

() => {
  class MyForm extends Component {
    constructor(props) {
      super(props);

      this.state = {
        caption: undefined,
        valid: false,
        validated: false,
        value: '',
        variant: undefined
      };

      this.handleTextInputChange = this.handleTextInputChange.bind(this);
      this.validate = this.validate.bind(this);
    }

    handleTextInputChange(event) {
      this.setState({
        value: event.target.value
      });
    }

    validate(event) {
      if (this.state.value) {
        this.setState({
          caption: 'Success, the value is valid.',
          valid: true,
          validated: true,
          variant: 'success'
        });
      } else {
        this.setState({
          caption: 'Oops, the value is invalid.',
          valid: false,
          validated: true,
          variant: 'danger'
        });
      }

      this.input.focus();
    }

    render() {
      const { caption, valid, validated, variant, value } = this.state;
      const formFieldProps = {
        input: TextInput,
        inputRef: ref => { this.input = ref; },
        invalid: validated ? !valid : undefined,
        label: 'Example',
        onChange: this.handleTextInputChange,
        caption,
        required: true,
        value,
        variant
      };

      return (
        <DemoLayout>
          <FormField {...formFieldProps} />
          <Button onClick={this.validate}>Validate</Button>
        </DemoLayout>
      );
    }
  }

  return <MyForm />;
}

Bidirectionality #

FormFields support right-to-left (RTL) languages.

<div dir="rtl">
  <ThemeProvider theme={{ direction: 'rtl' }}>
    <FormField
      input={TextInput}
      iconStart={<IconBackspace />}
      label="نموذج"
      defaultValue="مرحبا بالعالم"
      required
      requiredText="مطلوب" />
  </ThemeProvider>
</div>

API & Theme #

FormField Props #

The FormField component takes the following React props.

NameTypeDefaultDescription
captionstring | React$Element<*>

Caption associated with the input element; commonly used to provide help text

childrenReact$Node

Form input element

hideLabelboolean

Visually hide label, but keep available for assistive technologies

idstring

Id of the FormField

inputReact$ComponentType<*>

Form input class; alternative to children

labelstring | React$Element<*>required

Label associated with the input element

requiredboolean

Marks associated input as required

requiredTextstring | React$Element<*>'Required'

Text used to indicate a required field

rootPropsObject

Props to be applied directly to the root element, rather than the input

secondaryTextstring | React$Element<*>

Secondary text associated with the input element; takes precedence over requiredText

variant'danger' | 'success' | 'warning'

Available variants

Note

Unlike most other components, which apply undocumented properties to the root element, FormField applies undocumented properties, including as and css, to the input.

FormField Theme Variables #

These variables can be used as hooks to override this component's style at either a local or global level. The theme referenced below is whatever theme is available from props to the instance of this component.

VariableValue
FormFieldCaption_colortheme.color_mouse
FormFieldCaption_fontSizetheme.fontSize_mouse
FormFieldCaption_marginToptheme.space_stack_xxs
FormFieldCaption_marginTop_isGrouptheme.space_stack_sm
FormFieldLabel_colortheme.h6_color
FormFieldLabel_fontSizetheme.h6_fontSize
FormFieldLabel_fontWeighttheme.fontWeight_semiBold
FormFieldLabel_marginBottomtheme.space_stack_sm
FormFieldSecondaryText_fontSizetheme.fontSize_mouse
FormFieldSecondaryText_colortheme.color_mouse
FormFieldSecondaryText_color_requiredtheme.color_required
FormFieldSecondaryText_fontWeighttheme.fontWeight_regular

Usage #

When/How to Use #

Wrap each input in your app with a FormField for appropriate accessibility and styling.

Labels should be short (1-3 words) and indicate the type of input requested. Labels are not meant to explain the intent of the form as a whole.

Placeholder text is not a replacement for labels.

Do not use an asterisk in labels for required fields; use the required prop instead.

Error text should be short and concise, telling the user how to fix the problem.

Best Practices #

Wrap inputs in a FormField and provide a brief, descriptive label.

Use the appropriate variant to match your intent.

Don't use a variant that differs from intent, as this will cause confusion.

Provide useful secondaryText, when necessary.

Don't put too much secondaryText.

Provide a brief, helpful caption, when necessary.

Must be at least 8 characters, longer is better, any characters accepted.

Don't put too much text in a caption.

Passwords must contain at least 4 of the following:

  1. 8 or more characters
  2. less than 25 characters
  3. a letter
  4. a number
  5. a special character; not allowed: /\(),.

Use the hideLabel prop to hide the label in an accessible manner when the design requires it.