The table component can be a static or an interactive element. Some tables provide the possiblity to add, remove, edit a row or expand it for further information.

<dt-table [dataSource]="dataSource" dtSort #sortable>
  <dt-simple-text-column name="host" label="Host"></dt-simple-text-column>
  <dt-simple-number-column
    name="cpu"
    label="CPU"
    [formatter]="percentageFormatter"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="memoryPerc"
    label="Memory"
    [formatter]="percentageFormatter"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="memoryConsumption"
    label="Memory combined"
    [displayAccessor]="combineMemory"
    [sortAccessor]="memorySortAccessor"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="traffic"
    label="Traffic"
    [formatter]="trafficFormatter"
    sortable="false"
  ></dt-simple-number-column>
  <dt-header-row
    *dtHeaderRowDef="[
        'host',
        'cpu',
        'memoryPerc',
        'memoryConsumption',
        'traffic'
      ]"
  ></dt-header-row>
  <dt-row
    *dtRowDef="
        let row;
        columns: ['host', 'cpu', 'memoryPerc', 'memoryConsumption', 'traffic']
      "
  ></dt-row>
</dt-table>

The DtTable implementation enhances the Material's CDK table and also removes unneeded properties from the public API. The CDK table has a very different approach on how to define the table template, it does not use the native HTML table. Therefore, there are no td, tr or th tags involved. Instead, you need to define all possible columns that the table may show (depending on the data available) and then define which subset of columns you want to show in the table header and body by selecting from the column definitions.

Imports

You have to import the DtTableModule to use the dt-table. If you want to use the dt-expandable-cell component, Angular's BrowserAnimationsModule is required for animations. For more details on this see Step 2: Animations in the getting started guide. If you want to use the dt-order-cell, Angular's DragDropModule is required for reordering table rows via drag and drop. For more details on this see the Order section.

import { NgModule } from '@angular/core';
import { BrowserAnimationsModule } from '@angular/platform-browser/animations';
import { DtTableModule } from '@dynatrace/barista-components/table';

@NgModule({
  imports: [BrowserAnimationsModule, DtTableModule],
})
class MyModule {}

Table inputs

The DtTable component supports the following inputs. Find details about the usage of each input below.

Name Type Default Description
dataSource object[] | Observable | DataSource Data to be shown in the table.
loading boolean false Whether the table is loading or not.
multiExpand boolean false Whether the table allows multiple rows to be expanded at a time.
exportButton boolean | 'table' | 'visible' false Whether the table includes an export button.

Simple columns for basic use cases

For the most common column types (text and number columns) the Angular components provide a DtSimpleColumn implementation that wraps the underlying, more complex table setup. The dtSimpleColumn is an abstraction layer to the full fledged table implementation to make usage of recurring-patterns easier and to reduce boiler plate code that needs to be written to generate a simple table.

A dtTable using a dtSimpleColumn always needs the dtSort directive.

<dt-table [dataSource]="dataSource" dtSort #sortable>
  <dt-simple-text-column name="host" label="Host"></dt-simple-text-column>
  <dt-simple-number-column
    name="cpu"
    label="CPU"
    [formatter]="percentageFormatter"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="memoryPerc"
    label="Memory"
    [formatter]="percentageFormatter"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="memoryConsumption"
    label="Memory combined"
    [displayAccessor]="combineMemory"
    [sortAccessor]="memorySortAccessor"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="traffic"
    label="Traffic"
    [formatter]="trafficFormatter"
    sortable="false"
  ></dt-simple-number-column>
  <dt-header-row
    *dtHeaderRowDef="[
        'host',
        'cpu',
        'memoryPerc',
        'memoryConsumption',
        'traffic'
      ]"
  ></dt-header-row>
  <dt-row
    *dtRowDef="
        let row;
        columns: ['host', 'cpu', 'memoryPerc', 'memoryConsumption', 'traffic']
      "
  ></dt-row>
</dt-table>

DtSimpleColumn

The dtSimpleColumn provides a couple of inputs, which give the user easy access to certain functionality of the dtTable like sorting, formatting, and problem indicators.

Inputs

Name Type Default Description
name string - The name of the dtSimpleColumn refers to the name of the column which it uses to register itself at the table. This name can be used to reference the column in the dt-row. The name input is the only required input on the simple column.
label string - The label defines the string rendered into the header-cell of the given column. If no label is given, the name of the cell is being used. (Optional)
sortable boolean true The sortable input defines whether the column should be sortable.
displayAccessor<T> (data: T, name: string) => any - The displayAccessor function can be used to extract the displayable data from any row data. The name property passes the currently rendered column name, which can be used for more generic functions. If no displayAccessor is given, the dtSimpleColumn tries to extract a displayable data from rowData[name]. (Optional)
sortAccessor<T> (data: T, name: string) => string | number - The sortAccessor function can be used to extract sortable data from any row data. The name property passes the currently rendered column name, which can be used for more generic functions. This sortAccessor function will be used by the dataSource to access sortable values from the row data. (Optional)
formatter (displayValue: any) => string | DtFormattedValue - The formatter function can be used to fomat the displayed value, with either prepared DtFormatter functions or custom functions. Can be used on top of the displayAccessor function or standalone. The function gets passed either the output from the displayAccessor or the fallback data. (Optional)
hasProblem<T> (data: T, name: string) => DtIndicatorThemePalette - The hasProblem function can be used to evaluate if a cell should add the dtIndicator and if it should display error or warning. The function gets passed the row data and the name of the current column, which allows for more generic functions. The function needs to return either error or warning if a problem should be active. (Optional)
dtColumnProportion number - Exposes the dtColumnProportion of the dtCell directive for use with simple columns
comparator<T> (left: T, right: T, name: string) => number - The comparator function can be used to specify a custom comparator which is used to compare two rows. If this property is set, the sortAccessor for this column is ignored. This function must return < 0 if left is logically smaller than right, 0 if they are equivalent, otherwise > 0. (Optional)

Variants

Currently there are three predefined versions of the dtSimpleColumn exposed: dt-simple-number-column, dt-simple-text-column and dt-favorite-column. There are only small differences between the number and text column:

dt-simple-text-column
  • When sorting this column, the sort direction will start ascending, e.g. A -> Z
  • Column alignment is set to text -> left
dt-simple-number-column
  • When sorting this column, the sort direction will start descending, e.g. 100 -> 0
  • Column alignment is set to number -> right
  • When no formatter is given, the dtCount formatter will automatically be applied, e.g. 1000 -> 1k
dt-favorite-column
  • Column alignment is set to icon -> center
  • When sorting this column, the sort direction will start descending, e.g. ★ -> ☆

Default example of a favorite column:

<dt-table [dataSource]="dataSource" dtSort #sortable>
  <dt-favorite-column
    name="favorite"
    label="Favorite"
    (favoriteToggled)="toggleFavorite($event)"
  ></dt-favorite-column>
  <dt-simple-text-column name="host" label="Host"></dt-simple-text-column>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef dt-sort-header>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      {{ row.memoryPerc }} / {{ row.memoryTotal }}
    </dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['favorite', 'host', 'memory']"
  ></dt-header-row>
  <dt-row *dtRowDef="let row; columns: ['favorite', 'host', 'memory']"></dt-row>
</dt-table>

When no label is given, nothing will be rendered inside the header-cell:

<dt-table [dataSource]="dataSource">
  <dt-favorite-column
    name="favorite"
    sortable="false"
    (favoriteToggled)="toggleFavorite($event)"
  ></dt-favorite-column>
  <dt-simple-text-column
    name="host"
    label="Host"
    sortable="false"
  ></dt-simple-text-column>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      {{ row.memoryPerc }} / {{ row.memoryTotal }}
    </dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['favorite', 'host', 'memory']"
  ></dt-header-row>
  <dt-row *dtRowDef="let row; columns: ['favorite', 'host', 'memory']"></dt-row>
</dt-table>

Custom columns for advanced use cases

If the simple column implementation does not cover your use cases, you can use the underlying API to create your own column, cell, and header definitions as follows:

Each column definition is created with dt-header-cell and dt-cell inside an ng-container structural directive with a dtColumDef attribute directive applied to it.

<ng-container dtColumnDef="username">
  <dt-header-cell *dtHeaderCellDef>User name</dt-header-cell>
  <dt-cell *dtCellDef="let row"><span ngNonBindable>{{row.a}}</span></dt-cell>
</ng-container>

Note: dtCellDef not only exports the row data but also the same properties as *ngFor- using the same micro-syntax. The table header is defined with a dt-header-row component and a dtHeaderRowDef directive:

<dt-header-row *dtHeaderRowDef="['username', 'age', 'title']"></dt-header-row>

Finally the table row is defined with a dt-row component and a dtRowDef directive:

<dt-row *dtRowDef="let row; columns: ['username', 'age', 'title']"></dt-row>

Note: The dtRowDef also exports row context, which can be used for event and property bindings on the row element. See the source code of the examples on this page to see all the pieces in place.

<dt-table [dataSource]="data">
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      <dt-info-group>
        <dt-info-group-icon>
          <dt-icon name="host"></dt-icon>
        </dt-info-group-icon>
        <dt-info-group-title>{{ row.host }}</dt-info-group-title>
        Uptime: {{ row.uptime }}
      </dt-info-group>
    </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu | dtPercent }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ combineMemory(row) }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      {{ row.traffic | dtMegabytes: 1024 | dtRate: 's' }}
    </dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>

Components and attributes that define a table

Name Type Description
dtColumnDef Attribute Name for the column (to be used in the header and row definitions).
dtColumnAlign Attribute Sets the column alignment. Find possible values below in the "Alignment in tables" section.
dt-header-cell Directive Adds the appropriate classes (the generic dt-header-cell and the cell specific dt-column-css_friendly_column_name) and role (so the browser knows how to parse it. In this case it makes it act as a column header in a native html table)
*dtHeaderCellDef Attribute Captures the template of a column's header cell (the title for the column in the header) as well as cell-specific properties so the table can render it's header properly.
dt-cell Component Adds the appropriate classes and role (so the browser knows how to parse it. In this case it makes it act as a grid cell in a native html table).
*dtCellDef Attribute Exports the row data and the same properties as an ngFor in a way you can define what the cell should show. It also captures the template of the column's data row cell.
dt-header-row Component Placeholder for the header row. It is a container that contains the cell outlet. Adds the appropriate class and role.
*dtHeaderRowDef Attribute Defines the visible columns in the header out of all defined ones by receiving a columnName[].
dt-row Component Placeholder for the data rows. It is a container for the cell outlet. Adds the right class and role.
*dtRowDef Attribute Defines the visible columns in each row by receiving a columnName[] and also exposes the same micro-syntax that the dt-cell but for event and property binding.

Alignment in tables

Content can be centered, left- or right-aligned depending on the data. Header text should always follow the same alignment as the column content.

  • Left-aligned content
    • Text
    • Identification numbers beginning with letters, e.g. ID
  • Right-aligned content
    • Numbers
    • Date, time, year,...
    • IP addresses
  • Center-aligned content
    • Icons
    • Interactive components (e.g. switches)

Add the dtColumnAlign attribute to the ng-container that wraps the column definition to define the column's text alignment. The attribute accepts the following values:

Attribute values Alignment
left left
text left
id left
right right
number right
date right
ip right
center center
icon center
control center

Data source

The data source contains the data to be shown in the table. It can be an array, an observable holding an array or a DataSource object.

The Angular components provide a DtTableDataSource that provides a lot of functionality (like filtering, sorting, pagination) already set up. The DtTableOrderDataSource provides functionality to reorder the table, but does not work in combination with filtering, sorting or pagination. For use cases that can not be implemented using the DtTableDataSource you can always create your own data source that implements the DataSource interface.

Sorting

The DtSort and dt-sort-header are used to add sorting functionality to the table. To add sorting capabilities to a table add the dtSort directive to the dt-table component.

<dt-table ... dtSort ...></dt-table>

For each column that should be sortable by the user add dt-sort-header to the dt-header-cell. The dt-sort-header registers itself with the ID given to the dtColumnDef with the DtSort directive.

<dt-header-cell dt-sort-header ...></dt-header-cell>
<dt-table
  [dataSource]="dataSource"
  dtSort
  (dtSortChange)="sortData($event)"
  [dtSortDisabled]="disableSorting"
  [dtSortActive]="'cpu'"
  dtSortStart="asc"
  dtSortDirection="desc"
>
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell
      *dtHeaderCellDef
      dt-sort-header
      sort-aria-label="Change sort order for hosts"
    >
      Host
    </dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number">
    <dt-header-cell
      *dtHeaderCellDef
      dt-sort-header
      start="desc"
      sort-aria-label="Change sort order for CPUs"
    >
      CPU
    </dt-header-cell>
    <dt-cell *dtCellDef="let row"> {{ row.cpu | dtPercent }} </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell
      *dtHeaderCellDef
      dt-sort-header
      start="desc"
      sort-aria-label="Change sort order for memory"
    >
      Memory
    </dt-header-cell>
    <dt-cell *dtCellDef="let row">
      {{ row.memoryPerc | dtPercent }} of {{ row.memoryTotal | dtBytes }}
    </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell
      *dtHeaderCellDef
      disabled
      dt-sort-header
      sort-aria-label="Change sort order for network traffic"
    >
      Network traffic
    </dt-header-cell>
    <dt-cell *dtCellDef="let row">
      {{ row.traffic | dtBytes | dtRate: 's' }}
    </dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>
<button
  class="dt-toggle"
  dt-button
  variant="secondary"
  (click)="disableSorting = !disableSorting"
>
  Toggle disable sorting for all columns
</button>

DtSort

You can set the following inputs and outputs on the dtSort directive.

Inputs

Name Type Default Description
dtSortActive string The ID of the most recent active column.
dtSortDirection DtSortDirection asc The sort direction of the currently active colum, which can be used for initially sorting it.
dtSortDisabled boolean false Whether sorting is disabled for the entire table.
dtSortStart DtSortDirection Sort direction in which a column is initially sorted when the user interacts with it. May be overriden by the DtSortHeader's sort start.

Outputs

Name Type Default Description
sortChange EventEmitter<DtSortEvent> Event emmited when the user changes either the active sort or the sorting direction.

Methods

Name Description Parameters Return value
sort Sets the active sort ID and new sort direction sortable: DtSortHeader or active: string, direction: DtSortDirection void

DtSortHeader

You can set the following inputs and outputs on the dt-sort-header component.

Inputs

Name Type Default Description
disabled boolean Whether sorting is disabled for this sort header.
start DtSortDirection asc Overrides the sort start value of the containing DtSort.
sort-aria-label string Sets the aria label for the button used for sorting.

Accessibility

Please provide a sort-aria-label for each dt-sort-header to make the sorting experience accessible for all users. E.g. Change sort order for column hosts.

DtSortDirection

The type used for the sort direction, either asc or desc.

DtSortEvent

The event emitted when the user changes either the active sort or the sorting direction. The event contains the following properties.

Name Type Description
active string The ID of the currently active column.
direction DtSortDirection The direction for the currently active column.

Ordering

The DtOrder and dt-order-cell in combination with Angular's DragDropModule are used to add ordering functionality to the table. To add ordering capabilities to a table, import the DragDropModule and add the dtOrder and cdkDropList directive and cdkDropListData input to the dt-table component. When using the dt-order-cell, the data source provided for the table must be of type DtTableOrderDataSource. Tables using the dt-order-column can't be sorted or paginated.

<dt-table ... dtOrder cdkDropList [cdkDropListData]="dataSource" ...></dt-table>

The cdkDropListData gets the same data as the table's dataSource input. When combining the dt-order-cell with dt-simple-columns, their sortable input should be set to false. The DtTableOrderDataSource does not mutate the original data when ordering, if you want to persist the ordered state you have to take care of that yourself.

<dt-table
  [dataSource]="dataSource"
  dtOrder
  [dtOrderDisabled]="disableOrdering"
  cdkDropList
  [cdkDropListData]="dataSource"
>
  <ng-container dtColumnDef="order" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Order</dt-header-cell>
    <dt-order-cell *dtCellDef="let data; let index = index" [index]="index">
    </dt-order-cell>
  </ng-container>
  <dt-simple-text-column
    name="name"
    label="Rule name"
    sortable="false"
  ></dt-simple-text-column>
  <dt-header-row *dtHeaderRowDef="['order', 'name']"></dt-header-row>
  <dt-row *dtRowDef="let row; columns: ['order', 'name']"></dt-row>
</dt-table>
<button
  class="dt-toggle"
  dt-button
  variant="secondary"
  (click)="disableOrdering = !disableOrdering"
>
  Toggle disable ordering
</button>

DtOrder

You can set the following inputs and outputs on the dtOrder directive.

Inputs

Name Type Default Description
dtOrderDisabled boolean false Whether ordering is disabled for the entire table.

Outputs

Name Type Default Description
dtOrderChange EventEmitter<DtOrderChangeEvent> Event emmited when the user changes the order of the table, either by dragging a row to a new position or by changing the input value.

Methods

Name Description Parameters Return value
order Changes the position of a row based on the row's index currentIndex: number
targetIndex: number

Ordering with expandable rows

Ordering also works with custom columns and expandable rows.

<dt-table
  [dataSource]="dataSource"
  [multiExpand]="multiExpand"
  dtOrder
  cdkDropList
  [cdkDropListData]="dataSource"
>
  <ng-container [dtColumnDef]="'order'" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Order</dt-header-cell>
    <dt-order-cell *dtCellDef="let data; let index = index" [index]="index">
    </dt-order-cell>
  </ng-container>
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="details" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Details</dt-header-cell>
    <dt-expandable-cell
      *dtCellDef
      ariaLabel="Expand the row"
    ></dt-expandable-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['order', 'details', 'host', 'memory', 'cpu', 'traffic']"
  ></dt-header-row>
  <dt-expandable-row
    *dtRowDef="
      let row;
      columns: ['order', 'details', 'host', 'memory', 'cpu', 'traffic']
    "
    [expanded]="row.expanded"
  >
    <ng-template dtExpandableRowContent>
      Expandable section for {{ row.host }}
    </ng-template>
  </dt-expandable-row>
</dt-table>
<button
  class="dt-toggle"
  dt-button
  variant="secondary"
  (click)="multiExpand = !multiExpand"
>
  Toggle multiexpand (current value: {{ multiExpand }})
</button>

Ordering with data from observable

<button
  *ngIf="!isSubscribed; else cancel"
  class="dt-example-button"
  dt-button
  (click)="startSubscription()"
>
  Start subscription
</button>
<button
  class="dt-example-button"
  dt-button
  variant="secondary"
  (click)="clearRows()"
>
  Clear
</button>
<dt-table
  [dataSource]="dataSource"
  dtOrder
  cdkDropList
  [cdkDropListData]="dataSource"
>
  <ng-container [dtColumnDef]="'order'" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Order</dt-header-cell>
    <dt-order-cell *dtCellDef="let data; let index = index" [index]="index">
    </dt-order-cell>
  </ng-container>
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="control">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['order', 'host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['order', 'host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
  <dt-empty-state role="row">
    <dt-empty-state-item role="cell">
      <dt-empty-state-item-img>
        <img
          alt="glass"
          src=""
        />
      </dt-empty-state-item-img>
      <dt-empty-state-item-title aria-level="1">
        No data that matches your query
      </dt-empty-state-item-title>
      To add data to the query, click the start subscription button.
    </dt-empty-state-item>
  </dt-empty-state>
</dt-table>
<ng-template #cancel>
  <button class="example-button" dt-button (click)="cancelSubscription()">
    Cancel subscription
  </button>
</ng-template>

DtOrderEvent

The event emitted when the user changes the order of the table, either by dragging a row to a new position or by changing the input value. The event contains the following properties.

Name Type Description
previousIndex number The index of the row prior to changing the order.
currentIndex number The index of the row after changing the order.

Searching/Filtering

When tables contain large amounts of data, make it easier for the user to find entries by providing a search field above the table. The DtTableSearch component can be used in combination with a table to provide this functionality.

Filtering data for the table means filtering out rows that are passed via the data source. For highlighting the matched strings in the table the DtHighlight component can be used.

<dt-table-search
  name="tableSearch"
  [(ngModel)]="searchValue"
  placeholder="Search table data..."
  aria-label="Search table data"
></dt-table-search>
<dt-table [dataSource]="dataSource">
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      <dt-highlight [term]="searchValue">{{ row.host }}</dt-highlight>
    </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ combineMemory(row) }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      {{ row.traffic | dtMegabytes: 1024 | dtRate: 's' }}
    </dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>

DtTableSearch

Inputs

Name Type Default Description
value string '' The current search term.
placeholder string undefined Placeholder string for the input field (always needs to start with "Search ...").
aria-label string undefined An ARIA label describing the input field.
aria-labelledby string undefined A reference to an ARIA description of the input field.

Outputs

Name Type Description
valueChange EventEmitter<DtTableSearchChangeEvent> Event emitted when the user changes the search term.

Pagination and show more

The DtPagination component can be used in combination with a table to provide pagination for large datasets.

<dt-table [dataSource]="dataSource" dtSort #sortable>
  <dt-simple-text-column name="host" label="Host"></dt-simple-text-column>
  <dt-simple-number-column name="cpu" label="CPU"></dt-simple-number-column>
  <dt-simple-number-column
    name="memoryPerc"
    label="Memory"
    [formatter]="percentageFormatter"
    dtColumnProportion="3"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="memoryConsumption"
    label="Memory combined"
    [displayAccessor]="combineMemory"
    [sortAccessor]="memorySortAccessor"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="traffic"
    label="Traffic"
    sortable="false"
    [formatter]="trafficFormatter"
  ></dt-simple-number-column>
  <dt-header-row
    *dtHeaderRowDef="[
        'host',
        'cpu',
        'memoryPerc',
        'memoryConsumption',
        'traffic'
      ]"
  ></dt-header-row>
  <dt-row
    *dtRowDef="
        let row;
        columns: ['host', 'cpu', 'memoryPerc', 'memoryConsumption', 'traffic']
      "
  ></dt-row>
</dt-table>
<dt-pagination></dt-pagination>

In some use cases we might not know how much data we have in total. In this case a DtShowMore componenent might be more suitable.

<dt-table [dataSource]="dataSource">
  <dt-simple-text-column
    name="host"
    label="Host"
    sortable="false"
  ></dt-simple-text-column>
  <dt-simple-number-column
    name="cpu"
    label="CPU"
    sortable="false"
    [formatter]="percentageFormatter"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="memory"
    label="Memory"
    sortable="false"
    [formatter]="percentageFormatter"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="traffic"
    label="Traffic"
    sortable="false"
    [formatter]="trafficFormatter"
  ></dt-simple-number-column>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>
<button dt-show-more (click)="loadMore()">Show 5 more</button>

Selectable table rows

The DtTableRowSelector can be used for selectable rows in a table and the DtTableHeaderSelector to provide bulk selection in the column header.

The DtTableSelection directive has to be added to the table to enable selection handling. The directive also provides access to the currently selected rows and allows you to change selection programmatically. It also supports a selection limit by providing a DtTableSelectionConfig via the DT_TABLE_SELECTION_CONFIG injection token.

Inputs

Name Type Default Description
dtTableSelectionInitial T[] [] The rows that should be selected initially.
dtTableIsRowDisabled Predicate<T> () => false Predicate to disable matching rows

Outputs

Name Type Description
dtTableSelectionChange Observable<SelectionChange<T>> Event emitted when the selection changes either by user interaction or programmatically.
<dt-table
  [dataSource]="dataSource"
  dtSort
  dtTableSelection
  [dtTableIsRowDisabled]="isDisabled"
>
  <ng-container dtColumnDef="checkbox">
    <dt-table-header-selector
      *dtHeaderCellDef
      aria-label="Toggle all hosts"
    ></dt-table-header-selector>
    <dt-table-row-selector
      *dtCellDef="let row"
      [row]="row"
      [aria-label]="'Toggle ' + row.host"
    ></dt-table-row-selector>
  </ng-container>
  <dt-simple-text-column name="host" label="Host"></dt-simple-text-column>
  <dt-simple-number-column name="cpu" label="CPU"></dt-simple-number-column>
  <dt-header-row *dtHeaderRowDef="['checkbox', 'host', 'cpu']"></dt-header-row>
  <dt-row *dtRowDef="let row; columns: ['checkbox', 'host', 'cpu']"></dt-row>
</dt-table>
Current selection: {{ getCurrentSelection() }}

Sorting, paging and filtering out of the box with DtTableDataSource

Merging the streams for sorting, filtering and pagination can get quite tricky. The DtTableDataSource has all this features already set up.

Unified sorting (locale-aware for strings, null/undefined value treatment) is possible when using the DtTableDataSource. To do this, create a new DtTableDataSource instance with the data. You will have to connect the input instances for dtSort to the DtTableDataSource.

@Component({
  template: `
    <input dtInput (input)="updateFilter($event)" />
    <dt-table [dataSource]="dataSource" dtSort #sortable>
      [... column and row definitions]
    </dt-table>
    <dt-pagination></dt-pagination>
  `,
})
export class TableComponent implements OnInit {
  // Get the viewChild to pass the sorter, pagination, and filter reference to the datasource.
  @ViewChild('sortable', { read: DtSort, static: true }) sortable: DtSort;
  @ViewChild(DtPagination, { static: true }) pagination: DtPagination;

  // Create the Datasource instanciate it.
  dataSource: DtTableDataSource<object>;
  constructor() {
    this.dataSource = new DtTableDataSource(this.data);
  }

  ngOnInit(): void {
    // Set the dtSort reference on the dataSource, so it can react to sorting.
    this.dataSource.sort = this.sortable;
    // Set the dtPagination reference on the dataSource, so it can page the data.
    this.dataSource.pagination = this.pagination;
    // Set the pageSize to override the default page size.
    this.dataSource.pageSize = 10;
  }

  updateFilter(event: InputEvent): void {
    this.dataSource.filter = event.srcElement.value;
  }
}

The DtTableDataSource exposes two functions, that let the user define sort accessors for named columns. This enables the user to leverage DtTableDataSource sorting without the use of dt-simple-columns. The function addSortAccessorFunction(columnName: string, fn: DtSortAccessorFunction) can be used to add a custom sortAccessor function to the data source. The custom added sort accessors takes precedence over the automatically added ones from dt-simple-columns.

The function removeSortAccessorFunction(columnName: string) enables the user to remove a previously defined sortAccessor function.

export class TableComponent implements OnInit, OnDestroy {
  //...
  // Create the Datasource instanciate it.
  dataSource: DtTableDataSource<object>;
  constructor() {
    this.dataSource = new DtTableDataSource(this.data);
    this.dataSource.addSortAccessorFunction('memory', (row) => {
      return (row.memoryPerc / 100) * row.memoryTotal;
    });
  }

  ngOnDestroy() {
    // Not necessary, but possible
    this.dataSource.removeSortAccessorFunction('memory');
  }
}

The DtTableDataSource also exposes two functions, that let the user define custom comparators for named columns. This enables the user to leverage DtTableDataSource comparing without the use of dt-simple-columns. The function addComparatorFunction(columnName: string, fn: DtColumnComparatorFunction) can be used to add a custom comparator function to the data source. The custom added comparator takes precedence over defined sortAccessor functions.

The function removeComparatorFunction(columnName: string) enables the user to remove a previously defined comparator function.

<dt-table [dataSource]="dataSource" dtSort #sortable>
  <dt-simple-text-column
    name="host"
    label="Host"
    [comparator]="compareStringLengths"
  ></dt-simple-text-column>
  <dt-simple-text-column
    name="namespace"
    label="Namespace"
  ></dt-simple-text-column>
  <dt-header-row *dtHeaderRowDef="['host', 'namespace']"></dt-header-row>
  <dt-row *dtRowDef="let row; columns: ['host', 'namespace']"></dt-row>
</dt-table>

Expandable table rows

To show more details in context of a single table row, use expandable rows. They can be defined using the dt-expandable-row component. An expandable row has to contain a column with a details cell. A details cell can be added using dt-expandable-cell. The content inside the dt-expandable-row needs to be wrapped inside a ng-template with the dtExpandableRowContent directive to lazy load the content when the row is expanded.

<dt-table [dataSource]="dataSource" [multiExpand]="multiExpand">
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="control">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="details">
    <dt-header-cell *dtHeaderCellDef>Details</dt-header-cell>
    <dt-expandable-cell
      *dtCellDef
      ariaLabel="Expand the row"
    ></dt-expandable-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['details', 'host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-expandable-row
    *dtRowDef="
        let row;
        columns: ['details', 'host', 'cpu', 'memory', 'traffic']
      "
    [expanded]="row.expanded"
  >
    <ng-template dtExpandableRowContent>
      Expandable section for {{ row.host }}
    </ng-template>
  </dt-expandable-row>
</dt-table>
<button
  class="dt-toggle"
  dt-button
  variant="secondary"
  (click)="multiExpand = !multiExpand"
>
  Toggle multiexpand (current value: {{ multiExpand }})
</button>
Component/Attribute Type Description
dt-expandable-row Component Placeholder for the expandable data rows. It is a container for the cell outlet and an expandable section. Adds the right class and role.
dt-expandable-cell Component Adds the appropriate classes, role and content for the details cell in an expandable table.
ng-template dtExpandableContent Directive Wraps the content of the expandable row for lazy loading

Use the table's multiExpand input to allow multiple rows to be expanded at a time. The expanded state of an expandable row can be set programmatically by using the row's expanded input.

DtExpandableRow

Inputs

Name Type Default Description
expanded boolean false Whether the row is expanded or not.

Outputs

Name Type Description
expandChange EventEmitter<DtExpandableRowChangeEvent> Event emitted when the row's expandable state changes.
expanded EventEmitter<DtExpandableRow> Event emitted when the row is expanded.
collapsed EventEmitter<DtExpandableRow> Event emitted when the row is collapsed.

DtExpandableCell

Expandable rows have to contain one column definition which contains a dt-expandable-cell. Its aria-label or aria-labelledby input describes the action of toggling the state of the expandable row. A sample column definition for the details column could look like this:

<ng-container dtColumnDef="details" dtColumnAlign="control">
  <dt-header-cell *dtHeaderCellDef>Details</dt-header-cell>
  <dt-expandable-cell
    *dtCellDef
    ariaLabel="Expand the row"
  ></dt-expandable-cell>
</ng-container>

States

Empty state

When there is no data to display an empty state is shown. It can consist of some text that explains why there is no content and an illustration that helps to visualize the problem. Pass an empty state to the table using the <dt-empty-state> component or a custom empty state.

<dt-table [dataSource]="dataSource" dtSort [dtSortDisabled]="true">
  <dt-simple-text-column
    name="usersId"
    label="Users ID"
  ></dt-simple-text-column>
  <dt-simple-number-column
    name="sessionCount"
    label="Session count"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="averageDuration"
    label="Average duration"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="errors"
    label="Errors"
  ></dt-simple-number-column>
  <dt-simple-text-column name="country" label="Country"></dt-simple-text-column>
  <dt-simple-text-column name="city" label="City"></dt-simple-text-column>
  <dt-simple-text-column
    name="browserFamily"
    label="Browser family"
  ></dt-simple-text-column>
  <dt-simple-text-column name="device" label="Device"></dt-simple-text-column>
  <dt-empty-state role="row">
    <dt-empty-state-item role="cell">
      <dt-empty-state-item-img>
        <img
          alt="glass"
          src=""
        />
      </dt-empty-state-item-img>
      <!-- Give the empty state item an aria-level to indicate which level of heading it represents within the page -->
      <dt-empty-state-item-title aria-level="1">
        {{ emptyState.title }}
      </dt-empty-state-item-title>
      {{ emptyState.message }}
    </dt-empty-state-item>
  </dt-empty-state>
  <dt-header-row
    *dtHeaderRowDef="[
        'usersId',
        'sessionCount',
        'averageDuration',
        'errors',
        'country',
        'city',
        'browserFamily',
        'device'
      ]"
  ></dt-header-row>
  <dt-row
    *dtRowDef="
        let row;
        columns: [
          'usersId',
          'sessionCount',
          'averageDuration',
          'errors',
          'country',
          'city',
          'browserFamily',
          'device'
        ]
      "
  ></dt-row>
</dt-table>
<button
  class="dt-toggle"
  dt-button
  variant="secondary"
  (click)="toggleEmptyState()"
>
  Toggle empty state
</button>

Loading state

You can mark the table as loading using the table's loading input and pass the content to display with dtTableLoadingState directive.

<dt-table [dataSource]="dataSource" [loading]="tableLoading">
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <dt-loading-distractor dtTableLoadingState>
    Loading...
  </dt-loading-distractor>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>
<button
  class="dt-toggle"
  dt-button
  variant="secondary"
  (click)="tableLoading = !tableLoading"
>
  Toggle loading property
</button>

Problems: errors and warnings

If a table cell contains a problematic value, an indicator is used to highlight it with the error (red) status color. Warnings in tables indicate configuration issues using the warning (yellow) status color.

To get an error or warning indicator use the dtIndicator directive inside your dt-cell components on any HTML element or on your dt-cell component directly. The dt-row and dt-expandable-row will pick up if any dtIndicator was used inside the row's dt-cell and show the correct indicator. If one indicator has an error state set, the indicator on the row is an error indicator (i.e. error trumps warning). You can control the active state of the indicator by using the input named the same as dtIndicator.

<dt-cell [dtIndicator]="active" ...></dt-cell>

Single metric problem indication

If a single metric in a table entry is affected, only the affected value is highlighted.

Problem indication of entity or table row

If the problem affects the entity, but the affected metric is not in the table, the entity name is highlighted by the appropriate status color.

Note: Don't color hyperlinks even if the whole entity is affected, in this case use only the indicator on the left.

The example below shows both usages – a single metric inside a cell and the entire cell enhanced with the dtIndicator.

<dt-table [dataSource]="_dataSource">
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.name }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell
      [dtIndicator]="metricHasProblem(row, 'cpuUsage')"
      [dtIndicatorColor]="metricIndicatorColor(row, 'cpuUsage')"
      *dtCellDef="let row"
    >
      {{ row.cpuUsage | dtPercent }}
    </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      <span
        [dtIndicator]="metricHasProblem(row, 'memoryPerc')"
        [dtIndicatorColor]="metricIndicatorColor(row, 'memoryPerc')"
      >
        {{ row.memoryPerc | dtPercent }}
      </span>
      &nbsp;of&nbsp;
      <span
        [dtIndicator]="metricHasProblem(row, 'memoryTotal')"
        [dtIndicatorColor]="metricIndicatorColor(row, 'memoryTotal')"
      >
        {{ row.memoryTotal | dtBytes }}
      </span>
    </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      {{ row.traffic | dtBits | dtRate: 's' }}
    </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="empty">
    <dt-cell *dtCellDef="let row">This is empty</dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>
<button
  class="dt-toggle"
  dt-button
  variant="secondary"
  (click)="_toggleProblem()"
>
  Toggle problem
</button>

Advanced usage

To make the table header sticky, set the sticky input on the dtHeaderRowDef directive.

<dt-header-row
  *dtHeaderRowDef="['username', 'age', 'title']; sticky: true"
></dt-header-row>
<dt-table [dataSource]="dataSource">
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']; sticky: true"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>

Custom column width

The column width proportion can be customized using the dtColumnProportion attribute. It accepts a number input, e.g. [dtColumnProportion]="2" means that this column's width will be doubled compared to the regular ones.

<dt-table [dataSource]="dataSource">
  <ng-container dtColumnDef="host" dtColumnAlign="text" dtColumnProportion="3">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>

The column's minimum width can be set using the dtColumnMinWidth attribute. It accepts a CSS string describing the minimum width for the column. [dtColumnMinWidth]="200" means that this column's width will be at least 200px.

<dt-table [dataSource]="dataSource">
  <ng-container
    dtColumnDef="host"
    dtColumnAlign="text"
    [dtColumnMinWidth]="300"
  >
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number" dtColumnMinWidth="50%">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>

Dynamic columns

You can bind the column definitions to an array with a *ngFor directive.

<dt-table [dataSource]="dataSource">
  <ng-container
    *ngFor="let column of columnsDef"
    [dtColumnDef]="column.id"
    [dtColumnAlign]="column.type"
  >
    <dt-header-cell *dtHeaderCellDef>{{ column.title }}</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row[column.id] }}</dt-cell>
  </ng-container>
  <dt-header-row *dtHeaderRowDef="columnsName"></dt-header-row>
  <dt-row *dtRowDef="let row; columns: columnsName"></dt-row>
</dt-table>

Interactive rows

To apply a hover effect for the table rows add interactiveRows on the dt-table component.

<dt-table [dataSource]="dataSource" interactiveRows>
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
</dt-table>

Tables with two lines per row

If you want to have 2 lines of text and maybe an icon inside a row you can use the <dt-info-group> component. You can also take a look at the <dt-info-group> component's page for further information.

<dt-table [dataSource]="dataSource">
  <ng-container dtColumnDef="service">
    <dt-header-cell *dtHeaderCellDef>Service</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      <dt-info-group>
        <dt-info-group-icon>
          <dt-icon name="services"></dt-icon>
        </dt-info-group-icon>
        <dt-info-group-title>{{ row.service }}</dt-info-group-title>
        {{ row.location }}
      </dt-info-group>
    </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="responseTime" dtColumnAlign="right">
    <dt-header-cell *dtHeaderCellDef>Response time</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      {{ row.responseTime | dtCount }} ms
    </dt-cell>
  </ng-container>
  <ng-container dtColumnDef="failureRate" dtColumnAlign="right">
    <dt-header-cell *dtHeaderCellDef>Failure rate</dt-header-cell>
    <dt-cell *dtCellDef="let row"> {{ row.failureRate | dtPercent }} </dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['service', 'responseTime', 'failureRate']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['service', 'responseTime', 'failureRate']"
  ></dt-row>
</dt-table>

Responsive table

Since some tables might have a lot of data in them and screen space is very limited especially on devices with smaller screens you might want to switch between a table with expandable rows and normal rows. The example below shows a very simple approach how this might be done.

<dt-container-breakpoint-observer>
  <dt-table [dataSource]="dataSource">
    <ng-container dtColumnDef="host" dtColumnAlign="text">
      <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
      <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
    </ng-container>
    <ng-container dtColumnDef="cpu" dtColumnAlign="text">
      <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
      <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
    </ng-container>
    <ng-container dtColumnDef="memory" dtColumnAlign="number">
      <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
      <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
    </ng-container>
    <ng-container dtColumnDef="traffic" dtColumnAlign="control">
      <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
      <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
    </ng-container>
    <ng-container dtColumnDef="details" dtColumnAlign="text">
      <dt-header-cell *dtHeaderCellDef>Details</dt-header-cell>
      <dt-expandable-cell *dtCellDef></dt-expandable-cell>
    </ng-container>
    <dt-header-row *dtHeaderRowDef="_headerColumns"></dt-header-row>
    <dt-row
      *dtRowDef="
          let row;
          columns: ['host', 'cpu', 'memory', 'traffic'];
          let rowIndex = index;
          when: !_isTableNarrow
        "
    ></dt-row>
    <dt-expandable-row
      *dtRowDef="
          let row;
          columns: ['details', 'host', 'cpu'];
          let rowIndex = index;
          when: _isTableNarrow
        "
    >
      <ng-template dtExpandableRowContent>
        <dt-key-value-list>
          <dt-key-value-list-item>
            <dt-key-value-list-key>Memory</dt-key-value-list-key>
            <dt-key-value-list-value>{{ row.memory }}</dt-key-value-list-value>
          </dt-key-value-list-item>
          <dt-key-value-list-item>
            <dt-key-value-list-key>Traffic</dt-key-value-list-key>
            <dt-key-value-list-value>{{ row.traffic }}</dt-key-value-list-value>
          </dt-key-value-list-item>
        </dt-key-value-list>
      </ng-template>
    </dt-expandable-row>
  </dt-table>
</dt-container-breakpoint-observer>

Observable as data source

You can pass an observable to the [dataSource] property.

<button
  *ngIf="!isSubscribed; else cancel"
  class="dt-example-button"
  dt-button
  (click)="startSubscription()"
>
  Start subscription
</button>
<button
  class="dt-example-button"
  dt-button
  variant="secondary"
  (click)="clearRows()"
>
  Clear
</button>
<dt-table [dataSource]="dataSource">
  <ng-container dtColumnDef="host" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.host }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="cpu" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>CPU</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.cpu }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="memory" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef>Memory</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.memory }}</dt-cell>
  </ng-container>
  <ng-container dtColumnDef="traffic" dtColumnAlign="control">
    <dt-header-cell *dtHeaderCellDef>Network traffic</dt-header-cell>
    <dt-cell *dtCellDef="let row">{{ row.traffic }}</dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['host', 'cpu', 'memory', 'traffic']"
  ></dt-header-row>
  <dt-row
    *dtRowDef="let row; columns: ['host', 'cpu', 'memory', 'traffic']"
  ></dt-row>
  <dt-empty-state role="row">
    <dt-empty-state-item role="cell">
      <dt-empty-state-item-img>
        <img
          alt="glass"
          src=""
        />
      </dt-empty-state-item-img>
      <dt-empty-state-item-title aria-level="1">
        No data that matches your query
      </dt-empty-state-item-title>
      To add data to the query, click the start subscription button.
    </dt-empty-state-item>
  </dt-empty-state>
</dt-table>
<ng-template #cancel>
  <button class="example-button" dt-button (click)="cancelSubscription()">
    Cancel subscription
  </button>
</ng-template>

Extending DtSimpleColumn

You can extend from the simpleColumn to create your own predefined simplecolumn. To do this, simply extend from the DtSimpleColumnBase<T> abstract class and implement the deviation that suits your needs. A basic template for the simpleColumn could look like this (example from the dt-simple-number-column).

<ng-container [dtColumnDef]="name" dtColumnAlign="number">
  <ng-container *dtHeaderCellDef>
    <dt-header-cell *ngIf="!sortable">{{ label || name }}</dt-header-cell>
    <dt-header-cell *ngIf="sortable" dt-sort-header start="desc">
      {{ label || name }}
    </dt-header-cell>
  </ng-container>
  <dt-cell
    *dtCellDef="let data"
    [dtIndicator]="_getIndicator(data, name)"
    [dtIndicatorColor]="_getIndicator(data, name)"
  >
    {{ _getData(data) }}
  </dt-cell>
</ng-container>

Exporting

Simple

By setting the showExportButton input to true, an dtContextDialog button is added just below the table, or in line with pagination if present. Alternatively, you may set it to visible or table to only show those buttons. Developers should prefer true over the alternatives as it gives the user the most value; however, limiting the output may be desirable if there are, for example, many columns in the table data which should not be exported. This dialog can contain up to 2 buttons:

  • Export table data which triggers a download of the currently filtered data as shown, without regard for pagination.
  • Export visible data which triggers a download of the filtered data from the datasource and does not use a displayAccessor.

To exclude certain columns from export, set exportExcludeList to a string array of column names to exclude.

<dt-table
  [dataSource]="dataSource"
  dtSort
  #sortable
  [showExportButton]="true"
  [exportExcludeList]="['action']"
  [loading]="tableLoading"
>
  <ng-container dtColumnDef="host" dtColumnAlign="text" dtColumnProportion="3">
    <dt-header-cell *dtHeaderCellDef>Host</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      <dt-info-group>
        <dt-info-group-icon>
          <dt-icon name="host"></dt-icon>
        </dt-info-group-icon>
        <dt-info-group-title>{{ row.host }}</dt-info-group-title>
        Uptime: {{ row.uptime }}
      </dt-info-group>
    </dt-cell>
  </ng-container>
  <ng-container
    dtColumnDef="techType"
    dtColumnAlign="text"
    dtColumnProportion="1"
  >
    <dt-header-cell start="asc" *dtHeaderCellDef dt-sort-header i18n
      >Technology
    </dt-header-cell>
    <dt-cell *dtCellDef="let row"> {{techType(row)}} </dt-cell>
  </ng-container>
  <dt-simple-number-column name="cpu" label="CPU" dtColumnProportion="1">
  </dt-simple-number-column>
  <dt-simple-number-column
    name="memoryPerc"
    label="Memory"
    [formatter]="percentageFormatter"
    dtColumnProportion="1"
  >
  </dt-simple-number-column>
  <dt-simple-number-column
    name="memoryConsumption"
    label="Memory combined"
    [displayAccessor]="combineMemory"
    [sortAccessor]="memorySortAccessor"
    dtColumnProportion="2"
  ></dt-simple-number-column>
  <dt-simple-number-column
    name="traffic"
    label="Traffic"
    sortable="false"
    [formatter]="trafficFormatter"
    dtColumnProportion="2"
  >
  </dt-simple-number-column>
  <ng-container dtColumnDef="action" dtColumnAlign="text">
    <dt-header-cell *dtHeaderCellDef>Action</dt-header-cell>
    <dt-cell *dtCellDef="let row">
      <a href="/filter" dt-icon-button>
        <dt-icon variant="secondary" name="filter"></dt-icon>
      </a>
      <dt-context-dialog
        aria-label="Show more details"
        ariaLabelClose="Close context dialog"
      >
        <p>Drilldowns go here</p>
      </dt-context-dialog>
    </dt-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="[
        'host',
        'techType',
        'cpu',
        'memoryPerc',
        'memoryConsumption',
        'traffic',
        'action'
      ]"
  ></dt-header-row>
  <dt-row
    *dtRowDef="
        let row;
        columns: ['host', 'techType', 'cpu', 'memoryPerc', 'memoryConsumption', 'traffic', 'action']
      "
  ></dt-row>
</dt-table>
<dt-pagination></dt-pagination>

Selection

If dtTableSelection is enabled and you have connected dtTableSelection to dtTableDataSource (similar to dtSort), you also will have a third button:

  • Export selected rows which triggers a download of the display data, but just for selected rows.

Note: in this example showExportButton is set to visible to demonstrate the suppression of the Export table data button.

<dt-table
  [dataSource]="dataSource"
  dtSort
  dtTableSelection
  showExportButton="visible"
  [dtTableIsRowDisabled]="isDisabled"
  exportExcludeList="'details','checkbox'"
>
  <ng-container dtColumnDef="checkbox" dtColumnProportion="1">
    <dt-table-header-selector
      *dtHeaderCellDef
      aria-label="Toggle all hosts"
    ></dt-table-header-selector>
    <dt-table-row-selector
      *dtCellDef="let row"
      [row]="row"
      [aria-label]="'Toggle ' + row.host"
    ></dt-table-row-selector>
  </ng-container>
  <dt-simple-text-column
    name="host"
    label="Host"
    dtColumnProportion="2"
  ></dt-simple-text-column>
  <dt-simple-number-column
    name="cpu"
    label="CPU"
    dtColumnProportion="1"
  ></dt-simple-number-column>
  <ng-container dtColumnDef="details" dtColumnAlign="number">
    <dt-header-cell *dtHeaderCellDef dtColumnProportion="1"
      >Details</dt-header-cell
    >
    <dt-expandable-cell
      *dtCellDef
      ariaLabel="Expand the row"
    ></dt-expandable-cell>
  </ng-container>
  <dt-header-row
    *dtHeaderRowDef="['checkbox', 'details', 'host', 'cpu']"
  ></dt-header-row>
  <dt-expandable-row
    *dtRowDef="let row; columns: ['checkbox', 'details', 'host', 'cpu']"
  >
    <ng-template dtExpandableRowContent>
      Expandable section for {{ row.host }}
    </ng-template>
  </dt-expandable-row>
</dt-table>
Current selection: {{ getCurrentSelection() }}