The page navigation is complete. You may now navigate the page content as you wish.
Skip to main content

Layout::Grid

Added in v4.18.1

A component used to implement layouts based on CSS grid.

How to use this component

While it’s not necessary to be familiar with CSS3 grid specifications to use this component, some knowledge may be helpful in achieving more complex layouts.

A helpful reference with clear examples: CSS Tricks: Complete grid layout guide.

The Layout::Grid and optional Layout::Grid::Item components provide a way to quickly build out flexible grid-based layouts of components or elements without needing to write a lot of custom CSS code or understand all the intricacies of CSS grid styles.

Basic usage

The simplest way to implement a grid layout is by using the Layout::Grid component to wrap content directly. A grid layout of equal width “columns” is created by default.

Item 1
Item 2
Item 3
Item 4
<Hds::Layout::Grid>
  <Doc::Placeholder @height="40px" @text="Item 1" @background="#e4c5f3" />
  <Doc::Placeholder @height="40px" @text="Item 2" @background="#e5ffd2" />
  <Doc::Placeholder @height="40px" @text="Item 3" @background="#d2f4ff" />
  <Doc::Placeholder @height="40px" @text="Item 4" @background="#fff8d2" />
</Hds::Layout::Grid>

Every child of the grid container will be stretched to fit evenly within the underlying grid column tracks behaving as a grid item (for details on what this means, refer to the guide linked at the top of the page).

In some cases, it may be necessary to wrap one or more content items within the optional Layout::Grid::Item component. i.e., to group content together within a column or row, prevent content from being stretched to fit the underlying grid column width, or to make use of rowspan and colspan options in order to create more complex layouts. (See below for more details and examples on these features.)

Note: there is no strict need to use the Layout::Grid::Item subcomponent as a direct child of Layout::Grid; use it only when necessary to tweak grid styles of an individual child item such as via the @colspan/@rowspan arguments (to avoid rendering an extra Ember component).

Preventing content stretch

Wrap content in a Grid::Item to prevent it from stretching to fill the grid column.

Stretched badge
Non-stretched badge
<Hds::Layout::Grid @columnMinWidth="100%" @gap="16" as |LG|>
  <Hds::Badge @text="Stretched badge" @color="critical" />

  <LG.Item>
    <Hds::Badge @text="Non-stretched badge" @color="success" />
  </LG.Item>
</Hds::Layout::Grid>

Tag

To specify the HTML tag used to render the grid container and/or item(s), use the @tag argument.

<Hds::Layout::Grid @tag="ul" as |LG|>
  <li>{{! some content here }}</li>
  <LG.Item @tag="li">
    {{! some other content here }}
  </LG.Item>
  <li>{{! more content here }}</li>
</Hds::Layout::Grid>

While, by default, the component renders a <div>, we invite consumers to consider which semantic HTML tag is the correct one for the context in which the text is used to meet WCAG Success Criterion 1.3.1 Info and Relationships as the visual experience should match what is presented to the user with assistive technology.

Spacing

To control the spacing between grid items, use the @gap argument.

Pass a single value to set equal spacing between columns & rows.

Item 1
Item 2
Item 3
Item 4
<Hds::Layout::Grid @columnMinWidth="50%" @gap="16">
  <Doc::Placeholder @height="40px" @text="Item 1" @background="#e4c5f3" />
  <Doc::Placeholder @height="40px" @text="Item 2" @background="#e5ffd2" />
  <Doc::Placeholder @height="40px" @text="Item 3" @background="#d2f4ff" />
  <Doc::Placeholder @height="40px" @text="Item 4" @background="#fff8d2" />
</Hds::Layout::Grid>

To differentiate the vertical and horizontal spacing between items when they wrap on multiple rows, provide an array of two values to the @gap argument.

Item 1
Item 2
Item 3
Item 4
<Hds::Layout::Grid @columnMinWidth="50%" @gap={{array "16" "48"}}>
  <Doc::Placeholder @height="40px" @text="Item 1" @background="#e4c5f3" />
  <Doc::Placeholder @height="40px" @text="Item 2" @background="#e5ffd2" />
  <Doc::Placeholder @height="40px" @text="Item 3" @background="#d2f4ff" />
  <Doc::Placeholder @height="40px" @text="Item 4" @background="#fff8d2" />
</Hds::Layout::Grid>

The first value in the array refers to the vertical gap between “rows” of items (row-gap in CSS), the second one to the horizontal spacing between “columns” of items (column-gap in CSS).

The @gap argument accepts only pre-defined values, it can’t be used to provide custom spacing values. Refer to the Component API section for details on which values are accepted.

If you need to provide custom spacing values, see below how you can use a special escape hatch for this.

Important

The pre-defined value(s) passed to the @gap argument must be string(s), not numbers!

Non-standard gap values

If you absolutely have to use non-standard spacing value(s) for the grid gap, you can use the internal --hds-layout-grid-row-gap and --hds-layout-grid-column-gap CSS variables and pass custom values to them (e.g., via a local CSS variable or an inline style).

<Hds::Layout::Grid class="doc-grid-demo-custom-grid-column-gap">
  {{!
    // example of CSS code associated with the demo class:
    .doc-grid-demo-custom-grid-column-gap {
      --hds-layout-grid-column-gap: 13px;
    }
  }}
  {{!
    multiple grid items here, with a non-standard gap between them
  }}
</Hds::Layout::Grid>

In this case we’re overwriting only the “column” gap value via the custom CSS class.

If the grid items are wrapping on multiple lines, you have to overwrite both the “row” and “column” gap values.

<Hds::Layout::Grid
  {{style --hds-layout-grid-row-gap="10px" --hds-layout-grid-column-gap="0.625rem"}}
>
  {{!
    multiple grid items appearing on multiple rows
    with a vertical gap of 10px and a horizontal one of 0.625rem
  }}
</Hds::Layout::Grid>

Column min width

Specify a columnMinWidth size to exercise control over how many columns occupy a row. If the total widths of the columns add up to more than 100% of the parent they will automatically wrap to the next row as necessary to fit.

Note: The gap size will be subtracted from the columnMinWidth, so take this into account when specifying a column min width.

Using percentage values

Column min-widths specified as a percentage value will maintain the same size ratio in all browser screen widths.

Item 1
Item 2
Item 3
Item 4
<Hds::Layout::Grid @columnMinWidth="33.33%" @gap="16">
  <Doc::Placeholder @height="40px" @text="Item 1" @background="#e4c5f3" />
  <Doc::Placeholder @height="40px" @text="Item 2" @background="#e5ffd2" />
  <Doc::Placeholder @height="40px" @text="Item 3" @background="#d2f4ff" />
  <Doc::Placeholder @height="40px" @text="Item 4" @background="#fff8d2" />
</Hds::Layout::Grid>

Using fixed values

Column min-widths specified using pixels or other fixed units, allows you to create layouts which are “automatically” responsive.

Grid within a wider view

Narrow your browser window to see the responsive behavior.

Item 1
Item 2
Item 3
Item 4
<Hds::Layout::Grid @columnMinWidth="160px" @gap="16">
  <Doc::Placeholder @height="40px" @text="Item 1" @background="#e4c5f3" />
  <Doc::Placeholder @height="40px" @text="Item 2" @background="#e5ffd2" />
  <Doc::Placeholder @height="40px" @text="Item 3" @background="#d2f4ff" />
  <Doc::Placeholder @height="40px" @text="Item 4" @background="#fff8d2" />
</Hds::Layout::Grid>
The same grid within a narrower view

At the specified column min width, columns are forced to stack in this narrower view.

Item 1
Item 2
Item 3
Item 4
<div class="doc-grid-mobile-view">
  <Hds::Layout::Grid @columnMinWidth="160px" @gap="16">
    <Doc::Placeholder @height="40px" @text="Item 1" @background="#e4c5f3" />
    <Doc::Placeholder @height="40px" @text="Item 2" @background="#e5ffd2" />
    <Doc::Placeholder @height="40px" @text="Item 3" @background="#d2f4ff" />
    <Doc::Placeholder @height="40px" @text="Item 4" @background="#fff8d2" />
  </Hds::Layout::Grid>
</div>

Align

Use the @align argument to align grid items to the "start", "end", "center" or "stretch" them within the grid parent.

Note: The Grid parent will need a height set for the effect to be visible.

Item 1
Item 2
Item 3
Item 4
<div class="doc-grid-mobile-view">
  <Hds::Layout::Grid @columnMinWidth="50px" @gap="16" @align="center" {{style height="100%"}}>
    <Doc::Placeholder @height="40px" @text="Item 1" @background="#e4c5f3" />
  <Doc::Placeholder @height="40px" @text="Item 2" @background="#e5ffd2" />
  <Doc::Placeholder @height="40px" @text="Item 3" @background="#d2f4ff" />
  <Doc::Placeholder @height="40px" @text="Item 4" @background="#fff8d2" />
  </Hds::Layout::Grid>
</div>

Colspan & rowspan

Use the colspan and rowspan arguments of the Grid::Item component to set the number of columns or rows an item should occupy.

The following example has an underlying 4-column grid specified by setting a columnMinWidth of “25%”. It uses colspan and rowspan to create a flexible layout roughly resembling a typical web page layout.

Note: By default, if a height is set on the Grid parent, grid row heights will stretch proportionally to fill the Grid. To instead make a row conform to the minimum height of its content, you can pass an inline style as shown in the example.

Item 1
Item 2
Item 3
Item 4
<div {{style height="400px" border="1px solid"}}>
  <Hds::Layout::Grid 
    @columnMinWidth="25%" 
    @gap="12"
    {{style height="100%" grid-template-rows="min-content"}}
    as |LG|
  >
    <LG.Item @colspan={{4}}>
      <Doc::Placeholder @text="Item 1" @background="#e4c5f3" {{style padding="1em"}} />
    </LG.Item>

    <LG.Item @rowspan={{3}}>
      <Doc::Placeholder @height="100%" @text="Item 2" @background="#e5ffd2" />
    </LG.Item>

    <LG.Item @colspan={{3}}>
      <Doc::Placeholder @height="100%" @text="Item 3" @background="#d2f4ff" />
    </LG.Item>

    <LG.Item @colspan={{3}} @rowspan={{2}}>
      <Doc::Placeholder @height="100%" @text="Item 4" @background="#fff8d2" />
    </LG.Item>
  </Hds::Layout::Grid>
</div>

Common layout patterns

Below are examples of common layout patterns that can be achieved using the Layout::Grid component in combination with other HDS components.

Important

The examples below are meant to show how one could use the Layout::Grid component to implement certain common/standard UI patterns. They're not meant to be taken literally as they are and be used in production code.

Card layouts

Note: The following example makes use of nested Grid and Flex components to achieve its layout. This may be overkill in actual practice but demonstrates the possibilities for achieving layouts with just these layout components alone.

Basic 3-column layout

Active resources

5 active resources

There are 5 active resources inside this project.

View active resources

Card #2

Card #3

<Hds::Layout::Grid @columnMinWidth="33.33%" @gap="32">
  <Hds::Card::Container @level="mid" @hasBorder={{true}} {{style padding="24px"}}>
    <Hds::Layout::Grid @columnMinWidth="100%" @gap="16">
      <Hds::Layout::Flex @align="center" @gap="8">
        <Hds::IconTile @icon="cloud" @size="small" />
        <Hds::Text::Display @tag="h2" @size="300">
          Active resources
        </Hds::Text::Display>
      </Hds::Layout::Flex>
      <Hds::Layout::Grid @columnMinWidth="100%" @gap="8" as |LG|>
          <LG.Item>
            <Hds::Badge
              @text="5 active resources"
              @color="success"
              @icon="check-circle"
            />
          </LG.Item>
          <Hds::Text::Body @tag="p">
            There are 5 active resources inside this project.
          </Hds::Text::Body>
      </Hds::Layout::Grid>
      <Hds::Link::Standalone
        @icon="arrow-right"
        @iconPosition="trailing"
        @text="View active resources"
        @href="#"
      />
    </Hds::Layout::Grid>
  </Hds::Card::Container>

  <Hds::Card::Container @level="mid" @hasBorder={{true}} {{style padding="24px"}}>
    <Hds::Text::Display @tag="h2" @size="300">Card #2</Hds::Text::Display>
  </Hds::Card::Container>

  <Hds::Card::Container @level="mid" @hasBorder={{true}} {{style padding="24px"}}>
    <Hds::Text::Display @tag="h2" @size="300">Card #3</Hds::Text::Display>
  </Hds::Card::Container>
</Hds::Layout::Grid>

More complex layout

Wrap content with a Grid::Item as needed to achieve more complex layouts.

In Preview

Better together

HCP Terraform now works together with HCP Vault Secrets.

The combined solution enables your team to provision infrastructure with a scalable and least-privilege security approach for your secrets.

content

content

HCP Terraform Provider Resources

  • Deploy HCP Vault

    Integrate HCP Vault into your environment faster.

  • Deploy HCP Consul

    Manage your provisions and snapshot.

<Hds::Layout::Grid @columnMinWidth="33.33%" @gap="24" as |LG|>
  <LG.Item @colspan={{2}}>
    <Hds::Card::Container @level="mid" @hasBorder={{true}} {{style padding="24px"}} {{style background="radial-gradient(151.34% 168.34% at 0 0,#f6f9ff 0,#ebf2ff 100%)" }}>
      <Hds::Layout::Grid @columnMinWidth="100%" @gap="16" as |LG|>
        <LG.Item>
          <Hds::Badge @text="In Preview" @type="outlined" @color="highlight" />
        </LG.Item>
        <Hds::Text::Display @tag="h2" @size="300" @weight="bold">Better together</Hds::Text::Display>

        <Hds::Text::Body @tag="p" @weight="semibold">
          HCP Terraform now works together with HCP Vault Secrets.
        </Hds::Text::Body>
        <Hds::Text::Body @tag="p">
          The combined solution enables your team to provision infrastructure with a scalable and least-privilege security approach for your secrets.
        </Hds::Text::Body>
      </Hds::Layout::Grid>
    </Hds::Card::Container>
  </LG.Item>

  <Hds::Card::Container @level="mid" @hasBorder={{true}} {{style padding="24px"}}>
    <Hds::Text::Display @tag="h2" @size="300">content</Hds::Text::Display>
  </Hds::Card::Container>

  <Hds::Card::Container @level="mid" @hasBorder={{true}} {{style padding="24px"}}>
    <Hds::Text::Display @tag="h2" @size="300">content</Hds::Text::Display>
  </Hds::Card::Container>

  <LG.Item @colspan={{2}}>
    <Hds::Card::Container @level="mid" @hasBorder={{true}} {{style padding="24px"}}>
      <Hds::Layout::Grid @columnMinWidth="100%" @gap="16">
        <Hds::Text::Display @tag="h2" @size="300">HCP Terraform Provider Resources</Hds::Text::Display>
        <Hds::Layout::Grid @columnMinWidth="50%" @gap="24" @tag="ul" class="doc-grid-plain-list">
          <Hds::Layout::Grid @columnMinWidth="100%" @gap="8" @tag="li">
            <Hds::Text::Body @tag="p" @weight="semibold">Deploy HCP Vault</Hds::Text::Body>
            <Hds::Text::Body @tag="p">
              Integrate HCP Vault into your environment faster.
            </Hds::Text::Body>
          </Hds::Layout::Grid>
          <Hds::Layout::Grid @columnMinWidth="100%" @gap="8" @tag="li">
            <Hds::Text::Body @tag="p" @weight="semibold">Deploy HCP Consul</Hds::Text::Body>
            <Hds::Text::Body @tag="p">
              Manage your provisions and snapshot.
            </Hds::Text::Body>
          </Hds::Layout::Grid>
        </Hds::Layout::Grid>
      </Hds::Layout::Grid>
    </Hds::Card::Container>
  </LG.Item>
</Hds::Layout::Grid>

Component API

Layout::Grid

tag HTMLElementTagName
  • div (default)
A valid HTML tag name to be used to render the grid element.
columnMinWidth string
  • 0px (default)
Set any valid CSS dimension as a minimum width for the grid columns. If the total width of columns in a row exceeds 100% of the parent, columns will begin to wrap to the next row as necessary to fit. The column gap size is subtracted from this minimum width.

Note: With the default min-width of 0px, the columns will never wrap.
align enum
  • start
  • center
  • end
  • stretch
The value of the CSS align-items property, which controls the alignment of the grid items on the block axis within their grid areas (for a technical explanation: see MDN reference).

Note: we only expose a subset of the values allowed for this property, which cover the most common use cases.
gap enum
  • 4
  • 8
  • 12
  • 16
  • 32
  • 48
The gap spacing between rows and columns of the grid. Specify as either a single value or array of two values for setting the row and column spacing separately.
…attributes
This component supports use of ...attributes.

Contextual components

[LG].Item

The Layout::Grid::Item component, yielded as contextual component, to be used as child of the grid element to control its colspan/rowspan values (and other properties).

tag string
  • div (default)
HTML tag to be used to render the grid item element.
colspan number
The number of columns an item should span.
rowspan number
The number of rows an item should span.
yield
Elements passed as children are yielded as inner content of the element.
…attributes
This component supports use of ...attributes.

4.18.1

Added new Grid component