Form field

Overview for form-field

<mat-form-field> is a component used to wrap several Angular Material components and apply common Text field styles such as the underline, floating label, and hint messages.

In this document, "form field" refers to the wrapper component <mat-form-field> and "form field control" refers to the component that the <mat-form-field> is wrapping (e.g. the input, textarea, select, etc.)

The following Angular Material components are designed to work inside a <mat-form-field>:

Simple form field

The mat-form-field supports 4 different appearance variants which can be set via the appearance input. The fill an outline appearances implement the current version of the spec. The fill appearance displays the form field with a filled background box and an underline, while the outline appearance shows the form field with a border all the way around.

The legacy and standard appearances implement an older version of the spec where mat-form-field was shown with an underline beneath it. We recommend not using these appearances as they do not match the current spec.

There are a couple differences in behavior to be aware of between the different appearances.

We recommend that text prefix and suffixes in the fill and outline appearances only be used in conjunction with the floatLabel="always" option. This is because the resting label and the input value do not have the same alignment, and it is therefore impossible to align the prefix or suffix in a way that looks good when compared with both. In the standard and legacy appearances, the resting label and input value align, so this isn't an issue. We plan to improve support for text prefix and suffixes in the future so they can be used without floatLabel="always".

The second important difference is that the standard, fill, and outline appearances do not promote placeholders to labels. For the legacy appearance specifying <input placeholder="placeholder"> will result in a floating label being added to the mat-form-field. For the newer variants it will just add a normal placeholder to the input. If you want a floating label, add a <mat-label> to the mat-form-field instead.

Out of the box, if you do not specify an appearance for the <mat-form-field> it will default to legacy. However, this can be configured using a global provider to choose a different default appearance for your app.

@NgModule({
  providers: [
    {provide: MAT_FORM_FIELD_DEFAULT_OPTIONS, useValue: {appearance: 'fill'}}
  ]
})
Form field appearance variants

The floating label is a text label displayed on top of the form field control when the control does not contain any text or when <select matNativeControl> does not show any option text. By default, when text is present the floating label floats above the form field control. The label for a form field can be specified by adding a mat-label element.

In the legacy version of the <mat-form-field> (one that has no appearance attribute or has appearance="legacy") if a label is not specified, the placeholder attribute on the form control is promoted to a label. If a label is specified, the placeholder will be displayed as a normal placeholder. The placeholder will never be promoted to a label for standard, fill, and outline form fields. If you want to create a legacy form field with a placeholder but no label, you will need to specify an empty label to prevent the placeholder from being promoted.

<mat-form-field>
  <mat-label></mat-label>
  <input placeholder="Just a placeholder">
</mat-form-field>

If the form field control is marked with a required attribute, an asterisk will be appended to the label to indicate the fact that it is a required field. If unwanted, this can be disabled by setting the hideRequiredMarker property on <mat-form-field>

The floatLabel property of <mat-form-field> can be used to change this default floating behavior. It can set to never to hide the label instead of float it when text is present in the form field control. It can be set to always to float the label even when no text is present in the form field control. It can also be set to auto to restore the default behavior.

Form field with label

The floating label behavior can be adjusted globally by providing a value for MAT_FORM_FIELD_DEFAULT_OPTIONS in your application's root module. Like the floatLabel input, the option can be either set to always, never, or auto.

@NgModule({
  providers: [
    {provide: MAT_FORM_FIELD_DEFAULT_OPTIONS, useValue: {floatLabel: 'always'}}
  ]
})

Hint labels are additional descriptive text that appears below the form field's underline. A <mat-form-field> can have up to two hint labels; one start-aligned (left in an LTR language, right in RTL), and one end-aligned.

Hint labels are specified in one of two ways: either by using the hintLabel property of <mat-form-field>, or by adding a <mat-hint> element inside the form field. When adding a hint via the hintLabel property, it will be treated as the start hint. Hints added via the <mat-hint> hint element can be added to either side by setting the align property on <mat-hint> to either start or end. Attempting to add multiple hints to the same side will raise an error.

Form field with hints

Error messages can be shown under the form field underline by adding mat-error elements inside the form field. Errors are hidden initially and will be displayed on invalid form fields after the user has interacted with the element or the parent form has been submitted. Since the errors occupy the same space as the hints, the hints are hidden when the errors are shown.

If a form field can have more than one error state, it is up to the consumer to toggle which messages should be displayed. This can be done with CSS, ngIf or ngSwitch. Multiple error messages can be shown at the same time if desired, but the <mat-form-field> only reserves enough space to display one error message at a time. Ensuring that enough space is available to display multiple errors is up to the user.

Form field with error messages

Custom content can be included before and after the input tag, as a prefix or suffix. It will be included within the visual container that wraps the form control as per the Material specification.

Adding the matPrefix directive to an element inside the <mat-form-field> will designate it as the prefix. Similarly, adding matSuffix will designate it as the suffix.

Form field with prefix & suffix

In addition to the form field controls that Angular Material provides, it is possible to create custom form field controls that work with <mat-form-field> in the same way. For additional information on this see the guide on Creating Custom mat-form-field Controls.

<mat-form-field> has a color property which can be set to primary, accent, or warn. This will set the color of the form field underline and floating label based on the theme colors of your app.

<mat-form-field> inherits its font-size from its parent element. This can be overridden to an explicit size using CSS. We recommend a specificity of at least 1 element + 1 class.

mat-form-field.mat-form-field {
  font-size: 16px;
}
Form field theming

By itself, MatFormField does not apply any additional accessibility treatment to a control. However, several of the form field's optional features interact with the control contained within the form field.

When you provide a label via <mat-label>, MatFormField automatically associates this label with the field's control via a native <label> element, using the for attribute to reference the control's ID.

If a floating label is specified, it will be automatically used as the label for the form field control. If no floating label is specified, the user should label the form field control themselves using aria-label, aria-labelledby or <label for=...>.

When you provide informational text via <mat-hint> or <mat-error>, MatFormField automatically adds these elements' IDs to the control's aria-describedby attribute. Additionally, MatError applies aria-live="polite" by default such that assistive technology will announce errors when they appear.

This error occurs when you have specified two conflicting placeholders. Make sure that you haven't included both a placeholder property on your form field control and a <mat-placeholder> element. The <mat-placeholder> element is deprecated, you should use placeholder for placeholders and <mat-label> for labels.

This error occurs if you have added multiple hints for the same side. Keep in mind that the hintLabel property adds a hint to the start side.

This error occurs when you have not added a form field control to your form field. If your form field contains a native <input> or <textarea> element, make sure you've added the matInput directive to it and have imported MatInputModule. Other components that can act as a form field control include <mat-select>, <mat-chip-list>, and any custom form field controls you've created.

Indigo & Pink theme selected.