UNPKG

@c8y/ngx-components

Version:

Angular modules for Cumulocity IoT applications

196 lines 37.4 kB
import { Component, EventEmitter, forwardRef } from '@angular/core'; import { ActivatedRoute, Router } from '@angular/router'; import { AlertService, BuiltInActionType, DataGridService, gettext, ModalService, PRODUCT_EXPERIENCE_EVENT_SOURCE, Status } from '@c8y/ngx-components'; import { DeviceTypeGridColumn, RepositoryItemNameGridColumn, RepositoryService, RepositoryType } from '@c8y/ngx-components/repository/shared'; import { TranslateService } from '@ngx-translate/core'; import { cloneDeep } from 'lodash-es'; import { BsModalService } from 'ngx-bootstrap/modal'; import { from } from 'rxjs'; import { map } from 'rxjs/operators'; import { AddDeviceProfileComponent } from './add-device-profile.component'; import { PRODUCT_EXPERIENCE_DEVICE_PROFILE } from './device-profile.model'; import { DeviceProfileService } from './device-profile.service'; import * as i0 from "@angular/core"; import * as i1 from "@c8y/ngx-components/repository/shared"; import * as i2 from "@c8y/ngx-components"; import * as i3 from "ngx-bootstrap/modal"; import * as i4 from "@ngx-translate/core"; import * as i5 from "@angular/router"; import * as i6 from "./device-profile.service"; import * as i7 from "@angular/common"; export class DeviceProfileListComponent { constructor(repositoryService, gridService, modalService, bsModalService, translateService, alertService, router, activatedRoute, deviceProfileService) { this.repositoryService = repositoryService; this.gridService = gridService; this.modalService = modalService; this.bsModalService = bsModalService; this.translateService = translateService; this.alertService = alertService; this.router = router; this.activatedRoute = activatedRoute; this.deviceProfileService = deviceProfileService; this.sizeRequestDone = false; this.refresh$ = new EventEmitter(); this.isDataPresent$ = from(this.repositoryService.listRepositoryEntries(RepositoryType.PROFILE, { skipLegacy: true })).pipe(map(({ data }) => data.length > 0)); this.columns = [ new RepositoryItemNameGridColumn({ filterLabel: gettext('Filter device profile by name'), placeholder: gettext('ubuntu core') }), new DeviceTypeGridColumn({ filterLabel: gettext('Filter device profile by device type') }) ]; this.actionControls = []; this.pagination = { pageSize: 50, currentPage: 1 }; this.productExperienceEvent = { eventName: PRODUCT_EXPERIENCE_DEVICE_PROFILE.EVENTS.REPOSITORY, data: { component: PRODUCT_EXPERIENCE_DEVICE_PROFILE.COMPONENTS.DEVICE_PROFILE_LIST } }; this.noResultsMessage = gettext('No results to display.'); this.noDataMessage = gettext('No device profiles to display.'); this.noResultsSubtitle = gettext('Refine your search terms or check your spelling.'); this.noDataSubtitle = gettext('Add a new device profile by clicking below.'); this.serverSideDataCallback = this.onDataSourceModifier.bind(this); } ngOnInit() { this.actionControls.push({ type: BuiltInActionType.Edit, callback: this.editDeviceProfile.bind(this) }); this.actionControls.push({ type: 'duplicate', icon: 'copy', text: gettext('Duplicate'), callback: this.duplicateDeviceProfile.bind(this) }); this.actionControls.push({ type: BuiltInActionType.Delete, callback: this.deleteDeviceProfile.bind(this) }); } async onDataSourceModifier(dataSourceModifier) { const dataRequest = this.repositoryService.listRepositoryEntries(RepositoryType.PROFILE, { query: this.gridService.getQueryObj(dataSourceModifier.columns), skipDefaultOrder: true, params: { pageSize: dataSourceModifier.pagination.pageSize, currentPage: dataSourceModifier.pagination.currentPage } }); const filtererdSizeRequest = this.repositoryService .listRepositoryEntries(RepositoryType.PROFILE, { skipDefaultOrder: true, query: this.gridService.getQueryObj(dataSourceModifier.columns), params: { pageSize: 1 } }) .then(response => response?.paging?.totalPages); this.sizeRequest = this.repositoryService .listRepositoryEntries(RepositoryType.PROFILE, { skipDefaultOrder: true, params: { pageSize: 1 } }) .then(response => { this.sizeRequestDone = true; return response?.paging?.totalPages; }); const [dataResponse, size, filteredSize] = await Promise.all([ dataRequest, this.sizeRequest, filtererdSizeRequest ]); const { res, data, paging } = dataResponse; const serverSideDataResult = { res, data, paging, filteredSize, size }; return serverSideDataResult; } editDeviceProfile(deviceProfile) { this.router.navigate([deviceProfile.id], { relativeTo: this.activatedRoute }); } async createDeviceProfile() { const modal = this.bsModalService.show(AddDeviceProfileComponent, { class: 'modal-sm', ariaDescribedby: 'addDeviceProfileModalDescription', ariaLabelledBy: 'addDeviceProfileModalTitle', ignoreBackdropClick: true, keyboard: false, initialState: { productExperienceEvent: { ...this.productExperienceEvent, data: { ...this.productExperienceEvent.data, component: PRODUCT_EXPERIENCE_DEVICE_PROFILE.COMPONENTS.ADD_DEVICE_PROFILE } } } }).content; try { const profileId = await modal.result; modal.close(); this.router.navigateByUrl(`/device-profiles/${profileId}`); } catch (ex) { // do nothing } } async duplicateDeviceProfile(deviceProfile) { const copy = cloneDeep(deviceProfile); copy.id = null; copy.name = 'Duplicate of ' + deviceProfile.name; const mo = (await this.deviceProfileService.createDeviceProfile(copy)).data; this.router.navigateByUrl(`/device-profiles/${mo.id}`); } async deleteDeviceProfile(deviceProfile) { const deviceProfileName = deviceProfile.name; const title = gettext('Delete device profile'); const confirmationText = this.translateService.instant(gettext('You are about to delete a device profile "{{ deviceProfileName }}".'), { deviceProfileName }); const finalQuestion = this.translateService.instant(gettext('Do you want to proceed?')); try { await this.modalService.confirm(title, `${confirmationText} ${finalQuestion}`, Status.DANGER, { ok: gettext('Delete') }, {}, this.productExperienceEvent); await this.delete(deviceProfile.id); this.refresh$.next(); } catch (ex) { // do nothing } } trackByName(_index, column) { return column.name; } async delete(profileId) { try { await this.repositoryService.delete(profileId); this.alertService.success(gettext('Device profile deleted.')); } catch (ex) { this.alertService.addServerFailure(ex); } } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DeviceProfileListComponent, deps: [{ token: i1.RepositoryService }, { token: i2.DataGridService }, { token: i2.ModalService }, { token: i3.BsModalService }, { token: i4.TranslateService }, { token: i2.AlertService }, { token: i5.Router }, { token: i5.ActivatedRoute }, { token: i6.DeviceProfileService }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DeviceProfileListComponent, selector: "c8y-device-profile-list", providers: [ { provide: PRODUCT_EXPERIENCE_EVENT_SOURCE, useExisting: forwardRef(() => DeviceProfileListComponent) } ], ngImport: i0, template: "<c8y-title>{{ 'Device profiles' | translate }}</c8y-title>\n\n<c8y-breadcrumb>\n <c8y-breadcrumb-item\n icon=\"c8y-management\"\n label=\"{{ 'Management' | translate }}\"\n ></c8y-breadcrumb-item>\n <c8y-breadcrumb-item\n icon=\"c8y-device-profile\"\n label=\"{{ 'Device profiles' | translate }}\"\n ></c8y-breadcrumb-item>\n</c8y-breadcrumb>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n title=\"{{ 'Add device profile' | translate }}\"\n data-cy=\"device-profile-list--Add-device-profile\"\n (click)=\"createDeviceProfile()\"\n >\n <i c8yIcon=\"plus-circle\"></i>\n {{ 'Add device profile' | translate }}\n </button>\n</c8y-action-bar-item>\n\n<c8y-help\n src=\"/docs/device-management-application/managing-device-data/#managing-device-profiles\"\n></c8y-help>\n\n<div class=\"content-fullpage border-top border-bottom\">\n <c8y-data-grid\n [title]=\"'Device profiles' | translate\"\n [refresh]=\"refresh$\"\n [pagination]=\"pagination\"\n [columns]=\"columns\"\n [actionControls]=\"actionControls\"\n [infiniteScroll]=\"'auto'\"\n [serverSideDataCallback]=\"serverSideDataCallback\"\n >\n <c8y-ui-empty-state\n [icon]=\"stats?.size > 0 ? 'search' : 'c8y-tools'\"\n [title]=\"stats?.size > 0 ? (noResultsMessage | translate) : (noDataMessage | translate)\"\n [subtitle]=\"stats?.size > 0 ? (noResultsSubtitle | translate) : (noDataSubtitle | translate)\"\n *emptyStateContext=\"let stats\"\n [horizontal]=\"stats?.size > 0\"\n >\n <p *ngIf=\"stats?.size === 0\">\n <button\n class=\"btn btn-primary\"\n title=\"{{ 'Add device profile' | translate }}\"\n type=\"button\"\n (click)=\"createDeviceProfile()\"\n >\n {{ 'Add device profile' | translate }}\n </button>\n </p>\n </c8y-ui-empty-state>\n <ng-container *ngFor=\"let column of columns; trackBy: trackByName\">\n <c8y-column [name]=\"column.name\"></c8y-column>\n </ng-container>\n </c8y-data-grid>\n</div>\n", dependencies: [{ kind: "component", type: i2.ActionBarItemComponent, selector: "c8y-action-bar-item", inputs: ["placement", "priority", "itemClass", "injector", "groupId", "inGroupPriority"] }, { kind: "component", type: i2.BreadcrumbComponent, selector: "c8y-breadcrumb" }, { kind: "component", type: i2.BreadcrumbItemComponent, selector: "c8y-breadcrumb-item", inputs: ["icon", "translate", "label", "path", "injector"] }, { kind: "component", type: i2.EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "directive", type: i2.EmptyStateContextDirective, selector: "[emptyStateContext]" }, { kind: "directive", type: i2.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i7.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.ColumnDirective, selector: "c8y-column", inputs: ["name"] }, { kind: "component", type: i2.DataGridComponent, selector: "c8y-data-grid", inputs: ["title", "loadMoreItemsLabel", "loadingItemsLabel", "showSearch", "refresh", "columns", "rows", "pagination", "infiniteScroll", "serverSideDataCallback", "selectable", "singleSelection", "selectionPrimaryKey", "displayOptions", "actionControls", "bulkActionControls", "headerActionControls", "searchText", "configureColumnsEnabled", "showCounterWarning", "activeClassName", "expandableRows"], outputs: ["rowMouseOver", "rowMouseLeave", "rowClick", "onConfigChange", "onBeforeFilter", "onBeforeSearch", "onFilter", "itemsSelect", "onReload", "onAddCustomColumn", "onRemoveCustomColumn", "onColumnFilterReset", "onSort", "onPageSizeChange", "onColumnReordered", "onColumnVisibilityChange"] }, { kind: "component", type: i2.TitleComponent, selector: "c8y-title", inputs: ["pageTitleUpdate"] }, { kind: "component", type: i2.HelpComponent, selector: "c8y-help", inputs: ["src", "isCollapsed", "priority", "icon"] }, { kind: "pipe", type: i2.C8yTranslatePipe, name: "translate" }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DeviceProfileListComponent, decorators: [{ type: Component, args: [{ selector: 'c8y-device-profile-list', providers: [ { provide: PRODUCT_EXPERIENCE_EVENT_SOURCE, useExisting: forwardRef(() => DeviceProfileListComponent) } ], template: "<c8y-title>{{ 'Device profiles' | translate }}</c8y-title>\n\n<c8y-breadcrumb>\n <c8y-breadcrumb-item\n icon=\"c8y-management\"\n label=\"{{ 'Management' | translate }}\"\n ></c8y-breadcrumb-item>\n <c8y-breadcrumb-item\n icon=\"c8y-device-profile\"\n label=\"{{ 'Device profiles' | translate }}\"\n ></c8y-breadcrumb-item>\n</c8y-breadcrumb>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n title=\"{{ 'Add device profile' | translate }}\"\n data-cy=\"device-profile-list--Add-device-profile\"\n (click)=\"createDeviceProfile()\"\n >\n <i c8yIcon=\"plus-circle\"></i>\n {{ 'Add device profile' | translate }}\n </button>\n</c8y-action-bar-item>\n\n<c8y-help\n src=\"/docs/device-management-application/managing-device-data/#managing-device-profiles\"\n></c8y-help>\n\n<div class=\"content-fullpage border-top border-bottom\">\n <c8y-data-grid\n [title]=\"'Device profiles' | translate\"\n [refresh]=\"refresh$\"\n [pagination]=\"pagination\"\n [columns]=\"columns\"\n [actionControls]=\"actionControls\"\n [infiniteScroll]=\"'auto'\"\n [serverSideDataCallback]=\"serverSideDataCallback\"\n >\n <c8y-ui-empty-state\n [icon]=\"stats?.size > 0 ? 'search' : 'c8y-tools'\"\n [title]=\"stats?.size > 0 ? (noResultsMessage | translate) : (noDataMessage | translate)\"\n [subtitle]=\"stats?.size > 0 ? (noResultsSubtitle | translate) : (noDataSubtitle | translate)\"\n *emptyStateContext=\"let stats\"\n [horizontal]=\"stats?.size > 0\"\n >\n <p *ngIf=\"stats?.size === 0\">\n <button\n class=\"btn btn-primary\"\n title=\"{{ 'Add device profile' | translate }}\"\n type=\"button\"\n (click)=\"createDeviceProfile()\"\n >\n {{ 'Add device profile' | translate }}\n </button>\n </p>\n </c8y-ui-empty-state>\n <ng-container *ngFor=\"let column of columns; trackBy: trackByName\">\n <c8y-column [name]=\"column.name\"></c8y-column>\n </ng-container>\n </c8y-data-grid>\n</div>\n" }] }], ctorParameters: () => [{ type: i1.RepositoryService }, { type: i2.DataGridService }, { type: i2.ModalService }, { type: i3.BsModalService }, { type: i4.TranslateService }, { type: i2.AlertService }, { type: i5.Router }, { type: i5.ActivatedRoute }, { type: i6.DeviceProfileService }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"device-profile-list.component.js","sourceRoot":"","sources":["../../../device-profile/device-profile-list.component.ts","../../../device-profile/device-profile-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, forwardRef, OnInit } from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { IManagedObject, IResultList } from '@c8y/client';\nimport {\n  ActionControl,\n  AlertService,\n  BuiltInActionType,\n  Column,\n  DataGridService,\n  DataSourceModifier,\n  gettext,\n  ModalService,\n  PRODUCT_EXPERIENCE_EVENT_SOURCE,\n  ProductExperienceEvent,\n  ServerSideDataCallback,\n  ServerSideDataResult,\n  Status\n} from '@c8y/ngx-components';\nimport {\n  DeviceTypeGridColumn,\n  RepositoryItemNameGridColumn,\n  RepositoryService,\n  RepositoryType\n} from '@c8y/ngx-components/repository/shared';\nimport { TranslateService } from '@ngx-translate/core';\nimport { cloneDeep } from 'lodash-es';\nimport { BsModalService } from 'ngx-bootstrap/modal';\nimport { from, Observable } from 'rxjs';\nimport { map } from 'rxjs/operators';\nimport { AddDeviceProfileComponent } from './add-device-profile.component';\nimport { PRODUCT_EXPERIENCE_DEVICE_PROFILE } from './device-profile.model';\nimport { DeviceProfileService } from './device-profile.service';\n\n@Component({\n  selector: 'c8y-device-profile-list',\n  templateUrl: './device-profile-list.component.html',\n  providers: [\n    {\n      provide: PRODUCT_EXPERIENCE_EVENT_SOURCE,\n      useExisting: forwardRef(() => DeviceProfileListComponent)\n    }\n  ]\n})\nexport class DeviceProfileListComponent implements OnInit {\n  sizeRequest: Promise<number>;\n  sizeRequestDone = false;\n  refresh$: EventEmitter<void> = new EventEmitter();\n  isDataPresent$: Observable<boolean> = from(\n    this.repositoryService.listRepositoryEntries(RepositoryType.PROFILE, { skipLegacy: true })\n  ).pipe(map(({ data }) => data.length > 0));\n\n  columns: Column[] = [\n    new RepositoryItemNameGridColumn({\n      filterLabel: gettext('Filter device profile by name'),\n      placeholder: gettext('ubuntu core')\n    }),\n    new DeviceTypeGridColumn({ filterLabel: gettext('Filter device profile by device type') })\n  ];\n  actionControls: ActionControl[] = [];\n  serverSideDataCallback: ServerSideDataCallback;\n  pagination = {\n    pageSize: 50,\n    currentPage: 1\n  };\n  productExperienceEvent: ProductExperienceEvent = {\n    eventName: PRODUCT_EXPERIENCE_DEVICE_PROFILE.EVENTS.REPOSITORY,\n    data: {\n      component: PRODUCT_EXPERIENCE_DEVICE_PROFILE.COMPONENTS.DEVICE_PROFILE_LIST\n    }\n  };\n\n  noResultsMessage = gettext('No results to display.');\n  noDataMessage = gettext('No device profiles to display.');\n  noResultsSubtitle = gettext('Refine your search terms or check your spelling.');\n  noDataSubtitle = gettext('Add a new device profile by clicking below.');\n\n  constructor(\n    private repositoryService: RepositoryService,\n    private gridService: DataGridService,\n    private modalService: ModalService,\n    private bsModalService: BsModalService,\n    private translateService: TranslateService,\n    private alertService: AlertService,\n    private router: Router,\n    private activatedRoute: ActivatedRoute,\n    private deviceProfileService: DeviceProfileService\n  ) {\n    this.serverSideDataCallback = this.onDataSourceModifier.bind(this);\n  }\n\n  ngOnInit(): void {\n    this.actionControls.push({\n      type: BuiltInActionType.Edit,\n      callback: this.editDeviceProfile.bind(this)\n    });\n    this.actionControls.push({\n      type: 'duplicate',\n      icon: 'copy',\n      text: gettext('Duplicate'),\n      callback: this.duplicateDeviceProfile.bind(this)\n    });\n    this.actionControls.push({\n      type: BuiltInActionType.Delete,\n      callback: this.deleteDeviceProfile.bind(this)\n    });\n  }\n\n  async onDataSourceModifier(\n    dataSourceModifier: DataSourceModifier\n  ): Promise<ServerSideDataResult> {\n    const dataRequest: Promise<IResultList<IManagedObject>> =\n      this.repositoryService.listRepositoryEntries(RepositoryType.PROFILE, {\n        query: this.gridService.getQueryObj(dataSourceModifier.columns),\n        skipDefaultOrder: true,\n        params: {\n          pageSize: dataSourceModifier.pagination.pageSize,\n          currentPage: dataSourceModifier.pagination.currentPage\n        }\n      });\n\n    const filtererdSizeRequest: Promise<number> = this.repositoryService\n      .listRepositoryEntries(RepositoryType.PROFILE, {\n        skipDefaultOrder: true,\n        query: this.gridService.getQueryObj(dataSourceModifier.columns),\n        params: { pageSize: 1 }\n      })\n      .then(response => response?.paging?.totalPages);\n\n    this.sizeRequest = this.repositoryService\n      .listRepositoryEntries(RepositoryType.PROFILE, {\n        skipDefaultOrder: true,\n        params: { pageSize: 1 }\n      })\n      .then(response => {\n        this.sizeRequestDone = true;\n        return response?.paging?.totalPages;\n      });\n\n    const [dataResponse, size, filteredSize] = await Promise.all([\n      dataRequest,\n      this.sizeRequest,\n      filtererdSizeRequest\n    ]);\n\n    const { res, data, paging } = dataResponse;\n\n    const serverSideDataResult: ServerSideDataResult = {\n      res,\n      data,\n      paging,\n      filteredSize,\n      size\n    };\n\n    return serverSideDataResult;\n  }\n\n  editDeviceProfile(deviceProfile: Partial<IManagedObject>) {\n    this.router.navigate([deviceProfile.id], { relativeTo: this.activatedRoute });\n  }\n\n  async createDeviceProfile() {\n    const modal = this.bsModalService.show(AddDeviceProfileComponent, {\n      class: 'modal-sm',\n      ariaDescribedby: 'addDeviceProfileModalDescription',\n      ariaLabelledBy: 'addDeviceProfileModalTitle',\n      ignoreBackdropClick: true,\n      keyboard: false,\n      initialState: {\n        productExperienceEvent: {\n          ...this.productExperienceEvent,\n          data: {\n            ...this.productExperienceEvent.data,\n            component: PRODUCT_EXPERIENCE_DEVICE_PROFILE.COMPONENTS.ADD_DEVICE_PROFILE\n          }\n        }\n      }\n    }).content as AddDeviceProfileComponent;\n    try {\n      const profileId = await modal.result;\n      modal.close();\n      this.router.navigateByUrl(`/device-profiles/${profileId}`);\n    } catch (ex) {\n      // do nothing\n    }\n  }\n\n  async duplicateDeviceProfile(deviceProfile) {\n    const copy = cloneDeep(deviceProfile);\n    copy.id = null;\n    copy.name = 'Duplicate of ' + deviceProfile.name;\n    const mo = (await this.deviceProfileService.createDeviceProfile(copy)).data;\n    this.router.navigateByUrl(`/device-profiles/${mo.id}`);\n  }\n\n  async deleteDeviceProfile(deviceProfile) {\n    const deviceProfileName = deviceProfile.name;\n    const title = gettext('Delete device profile');\n    const confirmationText = this.translateService.instant(\n      gettext('You are about to delete a device profile \"{{ deviceProfileName }}\".'),\n      { deviceProfileName }\n    );\n    const finalQuestion = this.translateService.instant(gettext('Do you want to proceed?'));\n    try {\n      await this.modalService.confirm(\n        title,\n        `${confirmationText} ${finalQuestion}`,\n        Status.DANGER,\n        {\n          ok: gettext('Delete')\n        },\n        {},\n        this.productExperienceEvent\n      );\n      await this.delete(deviceProfile.id);\n      this.refresh$.next();\n    } catch (ex) {\n      // do nothing\n    }\n  }\n\n  trackByName(_index, column: Column): string {\n    return column.name;\n  }\n\n  private async delete(profileId) {\n    try {\n      await this.repositoryService.delete(profileId);\n      this.alertService.success(gettext('Device profile deleted.'));\n    } catch (ex) {\n      this.alertService.addServerFailure(ex);\n    }\n  }\n}\n","<c8y-title>{{ 'Device profiles' | translate }}</c8y-title>\n\n<c8y-breadcrumb>\n  <c8y-breadcrumb-item\n    icon=\"c8y-management\"\n    label=\"{{ 'Management' | translate }}\"\n  ></c8y-breadcrumb-item>\n  <c8y-breadcrumb-item\n    icon=\"c8y-device-profile\"\n    label=\"{{ 'Device profiles' | translate }}\"\n  ></c8y-breadcrumb-item>\n</c8y-breadcrumb>\n\n<c8y-action-bar-item [placement]=\"'right'\">\n  <button\n    class=\"btn btn-link\"\n    title=\"{{ 'Add device profile' | translate }}\"\n    data-cy=\"device-profile-list--Add-device-profile\"\n    (click)=\"createDeviceProfile()\"\n  >\n    <i c8yIcon=\"plus-circle\"></i>\n    {{ 'Add device profile' | translate }}\n  </button>\n</c8y-action-bar-item>\n\n<c8y-help\n  src=\"/docs/device-management-application/managing-device-data/#managing-device-profiles\"\n></c8y-help>\n\n<div class=\"content-fullpage border-top border-bottom\">\n  <c8y-data-grid\n    [title]=\"'Device profiles' | translate\"\n    [refresh]=\"refresh$\"\n    [pagination]=\"pagination\"\n    [columns]=\"columns\"\n    [actionControls]=\"actionControls\"\n    [infiniteScroll]=\"'auto'\"\n    [serverSideDataCallback]=\"serverSideDataCallback\"\n  >\n    <c8y-ui-empty-state\n      [icon]=\"stats?.size > 0 ? 'search' : 'c8y-tools'\"\n      [title]=\"stats?.size > 0 ? (noResultsMessage | translate) : (noDataMessage | translate)\"\n      [subtitle]=\"stats?.size > 0 ? (noResultsSubtitle | translate) : (noDataSubtitle | translate)\"\n      *emptyStateContext=\"let stats\"\n      [horizontal]=\"stats?.size > 0\"\n    >\n      <p *ngIf=\"stats?.size === 0\">\n        <button\n          class=\"btn btn-primary\"\n          title=\"{{ 'Add device profile' | translate }}\"\n          type=\"button\"\n          (click)=\"createDeviceProfile()\"\n        >\n          {{ 'Add device profile' | translate }}\n        </button>\n      </p>\n    </c8y-ui-empty-state>\n    <ng-container *ngFor=\"let column of columns; trackBy: trackByName\">\n      <c8y-column [name]=\"column.name\"></c8y-column>\n    </ng-container>\n  </c8y-data-grid>\n</div>\n"]}