thingsboard/ui-ngx/src/app/modules/home/components/attribute/attribute-table.component.html

283 lines
14 KiB
HTML
Raw Normal View History

2019-08-29 20:04:59 +03:00
<!--
2025-02-25 09:39:16 +02:00
Copyright © 2016-2025 The Thingsboard Authors
2019-08-29 20:04:59 +03:00
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
-->
2021-12-14 12:40:18 +02:00
<div class="mat-padding tb-entity-table tb-absolute-fill">
<div class="tb-entity-table-content tb-outlined-border flex flex-1 flex-col">
<mat-toolbar class="mat-mdc-table-toolbar" [class.!hidden]="mode !== 'default' || textSearchMode || !dataSource.selection.isEmpty()">
2019-08-29 20:04:59 +03:00
<div class="mat-toolbar-tools">
<div class="title-container flex flex-row items-center justify-start xs:flex-col xs:items-start xs:justify-center">
<span class="tb-entity-table-title">{{telemetryTypeTranslationsMap.get(attributeScope) | translate}}</span>
<mat-form-field class="mat-block tb-attribute-scope" *ngIf="!disableAttributeScopeSelection && !attributeScopeSelectionReadonly">
<mat-label translate>attribute.attributes-scope</mat-label>
<mat-select [disabled]="(isLoading$ | async)"
[ngModel]="attributeScope"
(ngModelChange)="attributeScopeChanged($event)">
<mat-option *ngFor="let scope of attributeScopes" [value]="scope">
{{ telemetryTypeTranslationsMap.get(toTelemetryTypeFunc(scope)) | translate }}
</mat-option>
</mat-select>
</mat-form-field>
</div>
<span class="flex-1"></span>
<button mat-icon-button
*ngIf="attributeScope !== attributeScopeTypes.CLIENT_SCOPE"
2019-08-29 20:04:59 +03:00
[disabled]="isLoading$ | async"
(click)="addAttribute($event)"
matTooltip="{{ 'action.add' | translate }}"
matTooltipPosition="above">
<mat-icon>add</mat-icon>
</button>
<button mat-icon-button
*ngIf="!isClientSideTelemetryTypeMap.get(attributeScope)"
2019-08-29 20:04:59 +03:00
[disabled]="isLoading$ | async"
(click)="reloadAttributes()"
matTooltip="{{ 'action.refresh' | translate }}"
matTooltipPosition="above">
<mat-icon>refresh</mat-icon>
</button>
<button mat-icon-button
2019-08-29 20:04:59 +03:00
[disabled]="isLoading$ | async"
(click)="enterFilterMode()"
matTooltip="{{ 'action.search' | translate }}"
matTooltipPosition="above">
<mat-icon>search</mat-icon>
</button>
</div>
</mat-toolbar>
<mat-toolbar class="mat-mdc-table-toolbar" [class.!hidden]="mode !== 'default' || !textSearchMode || !dataSource.selection.isEmpty()">
2019-08-29 20:04:59 +03:00
<div class="mat-toolbar-tools">
<button mat-icon-button
2019-08-29 20:04:59 +03:00
matTooltip="{{ 'action.search' | translate }}"
matTooltipPosition="above">
<mat-icon>search</mat-icon>
</button>
<mat-form-field class="flex-1">
2019-08-29 20:04:59 +03:00
<mat-label>&nbsp;</mat-label>
<input #searchInput matInput
[formControl]="textSearch"
2019-08-29 20:04:59 +03:00
placeholder="{{ 'common.enter-search' | translate }}"/>
</mat-form-field>
<button mat-icon-button (click)="exitFilterMode()"
2019-08-29 20:04:59 +03:00
matTooltip="{{ 'action.close' | translate }}"
matTooltipPosition="above">
<mat-icon>close</mat-icon>
</button>
</div>
</mat-toolbar>
<mat-toolbar class="mat-mdc-table-toolbar" color="primary" [class.!hidden]="mode !== 'default' || dataSource.selection.isEmpty()">
<div class="mat-toolbar-tools xs:flex xs:flex-row xs:flex-wrap">
<span class="tb-entity-table-info">
{{ (attributeScope === latestTelemetryTypes.LATEST_TELEMETRY ?
'attribute.selected-telemetry' : 'attribute.selected-attributes') | translate:{count: dataSource.selection.selected.length} }}
</span>
<span class="flex-1"></span>
<button [class.!hidden]="attributeScope === attributeScopeTypes.CLIENT_SCOPE"
class="button-widget-action"
mat-icon-button [disabled]="isLoading$ | async"
2019-08-29 20:04:59 +03:00
matTooltip="{{ 'action.delete' | translate }}"
matTooltipPosition="above"
2023-08-01 17:54:39 +03:00
(click)="deleteTelemetry($event)">
<mat-icon>delete</mat-icon>
</button>
<button mat-raised-button color="accent"
class="button-widget-action"
2019-08-29 20:04:59 +03:00
[disabled]="isLoading$ | async"
matTooltip="{{ 'attribute.show-on-widget' | translate }}"
matTooltipPosition="above"
(click)="enterWidgetMode()">
<mat-icon>now_widgets</mat-icon>
<span translate>attribute.show-on-widget</span>
</button>
</div>
</mat-toolbar>
<mat-toolbar class="mat-mdc-table-toolbar" color="primary" [class.!hidden]="mode !== 'widget'">
<div class="mat-toolbar-tools gap-2 xs:flex xs:flex-row xs:flex-wrap">
<div class="flex flex-1 flex-row items-stretch justify-start">
2019-08-29 20:04:59 +03:00
<span class="tb-details-subtitle">{{ 'widgets-bundle.current' | translate }}</span>
<tb-widgets-bundle-select class="flex-1"
style="margin-left: 5%"
required
[selectFirstBundle]="false"
[selectBundleAlias]="selectedWidgetsBundleAlias"
[ngModel]="null"
(ngModelChange)="onWidgetsBundleChanged($event)">
</tb-widgets-bundle-select>
2019-08-29 20:04:59 +03:00
</div>
<button mat-raised-button [class.!hidden]="!widgetsList.length"
2019-08-29 20:04:59 +03:00
color="accent"
class="button-widget-action"
2019-08-29 20:04:59 +03:00
[disabled]="isLoading$ | async"
matTooltip="{{ 'attribute.add-to-dashboard' | translate }}"
2019-08-29 20:04:59 +03:00
matTooltipPosition="above"
(click)="addWidgetToDashboard()">
<mat-icon>dashboard</mat-icon>
<span translate>attribute.add-to-dashboard</span>
</button>
<button mat-icon-button
2019-08-29 20:04:59 +03:00
[disabled]="isLoading$ | async"
matTooltip="{{ 'action.close' | translate }}"
matTooltipPosition="above"
(click)="exitWidgetMode()">
<mat-icon>close</mat-icon>
</button>
</div>
</mat-toolbar>
<div class="table-container flex-1" [class.!hidden]="mode === 'widget'">
<table mat-table [dataSource]="dataSource"
matSort [matSortActive]="pageLink.sortOrder.property" [matSortDirection]="pageLink.sortDirection()" matSortDisableClear>
2019-08-29 20:04:59 +03:00
<ng-container matColumnDef="select" sticky>
2023-05-19 11:45:41 +03:00
<mat-header-cell *matHeaderCellDef style="width: 40px;">
2019-08-29 20:04:59 +03:00
<mat-checkbox (change)="$event ? dataSource.masterToggle() : null"
[checked]="dataSource.selection.hasValue() && (dataSource.isAllSelected() | async)"
[indeterminate]="dataSource.selection.hasValue() && !(dataSource.isAllSelected() | async)">
</mat-checkbox>
</mat-header-cell>
<mat-cell *matCellDef="let attribute">
<mat-checkbox (click)="$event.stopPropagation()"
(change)="$event ? dataSource.selection.toggle(attribute) : null"
[checked]="dataSource.selection.isSelected(attribute)">
</mat-checkbox>
</mat-cell>
</ng-container>
<ng-container matColumnDef="lastUpdateTs">
2020-02-04 19:50:18 +02:00
<mat-header-cell *matHeaderCellDef mat-sort-header style="min-width: 150px; max-width: 150px; width: 150px;"> {{ 'attribute.last-update-time' | translate }} </mat-header-cell>
2019-08-29 20:04:59 +03:00
<mat-cell *matCellDef="let attribute">
{{ attribute.lastUpdateTs | date:'yyyy-MM-dd HH:mm:ss' }}
</mat-cell>
</ng-container>
<ng-container matColumnDef="key">
2020-02-04 19:50:18 +02:00
<mat-header-cell *matHeaderCellDef mat-sort-header style="width: 40%"> {{ 'attribute.key' | translate }} </mat-header-cell>
2023-05-02 09:19:53 +03:00
<mat-cell *matCellDef="let attribute" class="attribute">
<div class="flex flex-row" style="align-items: center">
2023-04-28 17:53:59 +03:00
<span class="ellipsis">{{ attribute.key }}</span>
<tb-copy-button
2023-05-01 16:26:49 +03:00
class="attribute-copy"
[disabled]="isLoading$ | async"
[copyText]="attribute.key"
tooltipText="{{ 'attribute.copy-key' | translate }}"
tooltipPosition="above"
icon="content_copy"
[style]="{padding: '4px', 'font-size': '16px', color: 'rgba(0,0,0,.87)'}"
>
</tb-copy-button>
</div>
2019-08-29 20:04:59 +03:00
</mat-cell>
</ng-container>
<ng-container matColumnDef="value">
2020-02-04 19:50:18 +02:00
<mat-header-cell *matHeaderCellDef style="width: 60%"> {{ 'attribute.value' | translate }} </mat-header-cell>
2019-08-29 20:04:59 +03:00
<mat-cell *matCellDef="let attribute"
2023-05-02 09:19:53 +03:00
class="tb-value-cell attribute"
2019-08-29 20:04:59 +03:00
(click)="editAttribute($event, attribute)">
<div class="flex flex-1 flex-row" style="align-items: center">
<div class="flex flex-1 flex-row" style="overflow: hidden; align-items: center;">
2023-04-28 17:53:59 +03:00
<span class="ellipsis">{{attribute.value | tbJson}}</span>
<tb-copy-button
*ngIf="attribute.value !== ''"
2023-05-01 16:26:49 +03:00
class="attribute-copy"
[disabled]="isLoading$ | async"
[copyText]="attribute.value | tbJson"
tooltipText="{{ 'attribute.copy-value' | translate }}"
tooltipPosition="above"
icon="content_copy"
[style]="{padding: '4px', 'font-size': '16px', color: 'rgba(0,0,0,.87)'}"
>
</tb-copy-button>
</div>
<span [class.!hidden]="isClientSideTelemetryTypeMap.get(attributeScope)">
2020-02-04 19:50:18 +02:00
<mat-icon>edit</mat-icon>
</span>
2023-08-04 17:49:52 +03:00
<button mat-icon-button *ngIf="attributeScope === latestTelemetryTypes.LATEST_TELEMETRY"
(click)="deleteTimeseries($event, attribute)">
<mat-icon>delete</mat-icon>
</button>
2020-02-04 19:50:18 +02:00
</div>
2019-08-29 20:04:59 +03:00
</mat-cell>
</ng-container>
<mat-header-row class="mat-row-select" *matHeaderRowDef="displayedColumns; sticky: true"></mat-header-row>
<mat-row class="mat-row-select" [class.mat-selected]="dataSource.selection.isSelected(attribute)"
2019-08-29 20:04:59 +03:00
*matRowDef="let attribute; columns: displayedColumns;" (click)="dataSource.selection.toggle(attribute)"></mat-row>
</table>
<span [class.!hidden]="(dataSource.isEmpty() | async) === false"
class="no-data-found flex items-center justify-center" translate>{{
2019-08-29 20:04:59 +03:00
attributeScope === latestTelemetryTypes.LATEST_TELEMETRY
? 'attribute.no-telemetry-text'
: 'attribute.no-attributes-text'
}}</span>
</div>
<mat-divider [class.!hidden]="mode === 'widget'"></mat-divider>
<mat-paginator [class.!hidden]="mode === 'widget'"
2019-08-29 20:04:59 +03:00
[length]="dataSource.total() | async"
[pageIndex]="pageLink.page"
[pageSize]="pageLink.pageSize"
[pageSizeOptions]="[10, 20, 30]"
[hidePageSize]="hidePageSize"
showFirstLastButtons></mat-paginator>
<ngx-hm-carousel *ngIf="mode === 'widget' && widgetsList.length > 0"
#carousel
[(ngModel)]="widgetsCarouselIndex"
2019-12-23 14:36:44 +02:00
(ngModelChange)="onWidgetsCarouselIndexChanged()"
[data]="widgetsList"
[infinite]="false"
class="carousel c-accent flex-1">
<section ngx-hm-carousel-container class="content">
<article class="item cursor-pointer"
ngx-hm-carousel-item
*ngFor="let widgets of widgetsList">
<tb-dashboard class="tb-absolute-fill"
[aliasController]="aliasController"
[widgets]="widgets"
[columns]="20"
[isEdit]="false"
[isMobileDisabled]="true"
[isEditActionEnabled]="false"
[isRemoveActionEnabled]="false">
</tb-dashboard>
</article>
</section>
<ng-template #carouselPrev>
<button mat-icon-button
*ngIf="widgetsCarouselIndex > 0"
matTooltip="{{ 'attribute.prev-widget' | translate }}"
matTooltipPosition="above">
<mat-icon>keyboard_arrow_left</mat-icon>
</button>
</ng-template>
<ng-template #carouselNext>
<button mat-icon-button
*ngIf="widgetsCarouselIndex < widgetsList.length - 1"
matTooltip="{{ 'attribute.next-widget' | translate }}"
matTooltipPosition="above">
<mat-icon>keyboard_arrow_right</mat-icon>
</button>
</ng-template>
<ng-template #carouselDot let-model>
<div class="ball"
[class.visible]="model.index === model.currentIndex"></div>
</ng-template>
</ngx-hm-carousel>
<span translate *ngIf="mode === 'widget' && widgetsLoaded &&
widgetsList.length === 0 &&
widgetBundleSet"
class="mat-headline-5 flex flex-1 items-center justify-center">widgets-bundle.empty</span>
<span translate *ngIf="mode === 'widget' && !widgetBundleSet"
class="mat-headline-5 flex flex-1 items-center justify-center">widget.select-widgets-bundle</span>
2019-08-29 20:04:59 +03:00
</div>
</div>