Documentation

Documentation versions (currently viewingVaadin 24)

Form Layout

Form Layout allows you to build responsive forms with multiple columns, and to position input labels above or to the side of the input.

Form Layout allows you to build responsive forms with multiple columns, and to position input labels above or to the side of the input.

Open in a
new tab
private responsiveSteps: FormLayoutResponsiveStep[] = [
  // Use one column by default
  { minWidth: 0, columns: 1 },
  // Use two columns, if layout's width exceeds 500px
  { minWidth: '500px', columns: 2 },
];

protected override render() {
  return html`
    <vaadin-form-layout .responsiveSteps="${this.responsiveSteps}">
      <vaadin-text-field label="First name"></vaadin-text-field>
      <vaadin-text-field label="Last name"></vaadin-text-field>
      <!-- Stretch the username field over 2 columns -->
      <vaadin-text-field colspan="2" label="Username"></vaadin-text-field>
      <vaadin-password-field label="Password"></vaadin-password-field>
      <vaadin-password-field label="Confirm password"></vaadin-password-field>
    </vaadin-form-layout>
  `;
}

Columns

By default, Form Layout has two columns, meaning it displays two input fields per line. When the layout width is smaller, it adjusts to a single-column.

Custom Layout

You can define how many columns Form Layout should use based on the screen width.

Important
Use the draggable split handle to resize Form Layout’s available space and to test its responsiveness.
Open in a
new tab
private responsiveSteps: FormLayoutResponsiveStep[] = [
  // Use one column by default
  { minWidth: 0, columns: 1 },
  // Use two columns, if the layout's width exceeds 320px
  { minWidth: '320px', columns: 2 },
  // Use three columns, if the layout's width exceeds 500px
  { minWidth: '500px', columns: 3 },
];

protected override render() {
  return html`
    <vaadin-split-layout>
      <vaadin-form-layout .responsiveSteps="${this.responsiveSteps}">
        <vaadin-text-field label="First name"></vaadin-text-field>
        <vaadin-text-field label="Last name"></vaadin-text-field>
        <vaadin-email-field label="Email"></vaadin-email-field>
      </vaadin-form-layout>
      <div></div>
    </vaadin-split-layout>
  `;
}

A single-column layout is preferable to a multi-column one. A multi-column layout can be prone to confusion and misinterpretation by the user.

However, related fields can be placed in a line without confusion, typically. Examples of this would be first and last name, address fields such as postal code and city, and ranged input for dates, time, and currency.

Column Span

When using a multi-column layout, you can define a colspan for each component. The colspan determines how many columns a component extends or stretches across.

For example, if you have a Form Layout with three columns and a component’s colspan is set to 3, it takes up the entire width of the Form Layout.

Open in a
new tab
<vaadin-text-field label="Title" colspan="3"></vaadin-text-field>

Label Position

Input fields' built-in labels are positioned above the input. Form Layout supports side-positioned labels, provided they are wrapped in Form Items and the label position is set to aside.

The only reason for wrapping labels in Form Items is to put the labels to the side of the input.

Top

Users complete forms that have top-positioned labels more quickly because they provide a consistent scanning pattern — top-down, as opposed to zigzag — while minimizing the distance between the label and input.

Top-positioned labels are also less prone to causing layout issues due to variable label lengths, which happens usually in multilingual applications. However, they do result in vertically longer forms. This is why sectioning is important.

Side

Side-positioned labels help reduce a form’s total height. This is especially useful for longer forms and when vertical space is limited.

Labels positioned on the side are also often used when there is a need to compare numeric data.

Open in a
new tab
<vaadin-form-layout>
  <!-- Wrap fields into form items, which
       displays labels on the side by default -->
  <vaadin-form-item>
    <label slot="label">Revenue</label>
    <vaadin-text-field>
      <span slot="suffix">EUR</span>
    </vaadin-text-field>
  </vaadin-form-item>
  <vaadin-form-item>
    <label slot="label">Expenses</label>
    <vaadin-text-field>
      <span slot="suffix">EUR</span>
    </vaadin-text-field>
  </vaadin-form-item>
  <vaadin-form-item>
    <label slot="label">Invoices</label>
    <vaadin-text-field>
      <span slot="suffix">EUR</span>
    </vaadin-text-field>
  </vaadin-form-item>
</vaadin-form-layout>

Aim for similar-length labels to keep the distance between the labels and input fields consistent. Inconsistent spacing can slow the user in completing the form.

Forms that use this position require more horizontal space, which isn’t always ideal in narrow forms. It’s recommended to configure Form Layout to use top-positioned labels when the form has a narrow width.

Responsive Label Position

Similar to the number of columns, the label position is configurable based on the width of the layout. For example, you can position the labels to the side when there is ample horizontal space available, and on top for narrower screens.

Native Input Fields

Form Item allows you to set a label for any type of component that you want to use in a Form Layout. It supports both Vaadin components and native HTML components.

Open in a
new tab
<vaadin-form-layout>
  <vaadin-form-item>
    <label slot="label">Revenue</label>
    <input type="text" />
  </vaadin-form-item>
</vaadin-form-layout>

Multiple Fields

Form Item only supports placing a single field inside. Where you need to place multiple fields, Custom Field should be used as a wrapper:

Open in a
new tab
@query('#month-field')
private accessor monthField!: Select;

@query('#year-field')
private accessor yearField!: Select;

@state()
private accessor months = Array.from({ length: 12 }, (_, i) => `${i + 1}`.padStart(2, '0'));

@state()
private accessor years = Array.from({ length: 11 }, (_, i) => `${i + new Date().getFullYear()}`);

protected override firstUpdated() {
  // Set title for screen readers
  this.monthField.focusElement!.setAttribute('title', 'Month');
  this.yearField.focusElement!.setAttribute('title', 'Year');
}

render() {
  return html`
    <vaadin-form-layout>
      <vaadin-form-item>
        <label slot="label">Expiration</label>
        <vaadin-custom-field
          .parseValue="${(value: string) => {
            return value ? value.split('/') : ['', ''];
          }}"
          .formatValue="${(values: unknown[]) => {
            return values[0] && values[1] ? values.join('/') : '';
          }}"
        >
          <vaadin-horizontal-layout theme="spacing-xs">
            <vaadin-select
              id="month-field"
              placeholder="Month"
              .items="${this.months.map((month) => ({ label: month, value: month }))}"
            ></vaadin-select>
            <vaadin-select
              id="year-field"
              placeholder="Year"
              .items="${this.years.map((year) => ({ label: year, value: year }))}"
            ></vaadin-select>
          </vaadin-horizontal-layout>
        </vaadin-custom-field>
      </vaadin-form-item>
    </vaadin-form-layout>
  `;
}

Keeping fields in individual Form Items, however, is preferable. Wrapped fields can be hard to distinguish visually since they usually have no individual label except for a placeholder, which is only visible when the field has no value.

Best Practices

Sectioning

Longer forms should be split into smaller, more manageable and user-friendly sections using sub-headings, Tabs, Details or separate views when possible. Each section should consist of related content and fields.

Button Placement

Use the following guidelines for Button placement in forms:

  • Buttons should be placed below the form with which they are associated.

  • Buttons should be aligned to the left.

  • Primary action should be placed first, followed by other actions, in order of importance.

For more information, see the Button documentation.

7B8E4F98-540C-4622-A39F-907C95E9DFFD