@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
194 lines • 51.9 kB
JavaScript
import { Component, forwardRef, Input } from '@angular/core';
import { NG_VALUE_ACCESSOR } from '@angular/forms';
import { gettext } from '@c8y/ngx-components';
import { BehaviorSubject, combineLatest, from } from 'rxjs';
import { debounceTime, distinctUntilChanged, map, shareReplay, switchMap, tap } from 'rxjs/operators';
import { DatapointLibraryService } from './datapoint-library.service';
import { AddButtonTypes } from './datapoint-selector-list-item/datapoint-selector-list-item.component';
import * as i0 from "@angular/core";
import * as i1 from "./datapoint-library.service";
import * as i2 from "@c8y/ngx-components/assets-navigator";
import * as i3 from "@c8y/ngx-components";
import * as i4 from "@angular/common";
import * as i5 from "@angular/forms";
import * as i6 from "./datapoint-selector-list-item/datapoint-selector-list-item.component";
import * as i7 from "./pipes/includes-datapoint.pipe";
import * as i8 from "./pipes/datapoint-label.pipe";
export class DatapointSelectorComponent {
constructor(datapointService) {
this.datapointService = datapointService;
this.allowChangingContext = true;
this.allowDatapointsFromMultipleAssets = true;
this.selectedDatapoints = new Array();
this.defaultActiveState = true;
this.ignoreDatapointTemplates = false;
this.datapointTemplatesOnly = false;
this.guessDatapointUnit = true;
this.allowSearch = true;
this.hideSelection = false;
this.itemsEditable = true;
this.searchString = '';
this.maxNumberOfDatapoints = 50;
this.AddButtonTypes = AddButtonTypes;
this.loadingDatapoints = false;
this.assetSelection = new BehaviorSubject(null);
this.emptyStateSubtitleWhenNoMatchingDataPoints = gettext('Try another search term.');
this.emptyStateSubtitleWhenNoDataPointsInAsset = gettext('Select an asset with data points from the list.');
this.searchString$ = new BehaviorSubject('');
this.touched = false;
}
ngOnInit() {
this.selectorTitle = this.datapointTemplatesOnly
? gettext('Available data point templates')
: gettext('Available data points');
this.emptyStateTitle = this.datapointTemplatesOnly
? gettext('No data point templates to display.')
: gettext('No data points to display.');
this.selectedListTitle = this.datapointTemplatesOnly
? gettext('Selected data point templates')
: gettext('Selected data points');
this.setupObservables();
if (!this.ignoreDatapointTemplates) {
this.datapointLibraryEntries = from(this.datapointService.getFirstDatapointLibraryPage());
}
if (this.contextAsset) {
this.selectionChanged(this.contextAsset);
}
}
writeValue(obj) {
this.selectedDatapoints = obj;
}
registerOnChange(fn) {
this.onChange = fn;
}
registerOnTouched(fn) {
this.onTouched = fn;
}
datapointAdded(dp) {
this.markAsTouched();
dp.__active = this.defaultActiveState;
if (this.guessDatapointUnit && !dp.unit) {
this.datapointService.guessUnitOfDatapoint(dp.fragment, dp.series, dp.__target).then(unit => {
dp.unit = unit;
});
}
this.selectedDatapoints = [...this.selectedDatapoints, dp];
this.emitCurrentSelection();
}
datapointRemoved(dp) {
this.markAsTouched();
this.selectedDatapoints = this.selectedDatapoints.filter(tmp => tmp.fragment !== dp.fragment ||
tmp.series !== dp.series ||
tmp.__target?.id !== dp.__target?.id);
this.emitCurrentSelection();
}
selectionChanged(evt) {
if (Array.isArray(evt) && evt.length !== 0) {
return this.selectAsset(evt[0]);
}
if (!Array.isArray(evt) && evt.items) {
return this.selectionChanged(evt.items);
}
if (!Array.isArray(evt) && evt.id) {
return this.selectAsset(evt);
}
// reset selection
this.assetSelection.next(null);
}
trackByFn(_index, item) {
return `${item.fragment}-${item.__target?.id}-${item.series}`;
}
searchStringChanged(newValue = '') {
this.searchString$.next(newValue);
this.searchString = newValue;
}
setupObservables() {
this.datapoints$ = this.assetSelection.pipe(tap(() => {
this.loadingDatapoints = true;
}), switchMap(asset => asset?.id
? this.datapointService.getDatapointsOfAsset(asset, this.ignoreDatapointTemplates, this.datapointTemplatesOnly)
: []), tap(() => (this.loadingDatapoints = false)), shareReplay(1));
this.searchStringChanges$ = this.searchString$.pipe(distinctUntilChanged(), debounceTime(500), shareReplay(1));
this.filteredDatapoints$ = combineLatest([this.searchStringChanges$, this.datapoints$]).pipe(map(([searchString, datapoints]) => {
if (!searchString) {
return datapoints;
}
const lowerCaseSearchString = searchString.toLowerCase();
return datapoints.filter(datapoint => this.includesSearchString(datapoint, lowerCaseSearchString));
}), map(filtered => filtered.slice(0, this.maxNumberOfDatapoints)));
}
selectAsset(asset) {
this.assetSelection.next(asset);
this.searchStringChanged();
if (!this.allowDatapointsFromMultipleAssets) {
this.clearSelection();
}
}
clearSelection() {
this.selectedDatapoints = [];
this.emitCurrentSelection();
}
emitCurrentSelection() {
this.onChange(this.selectedDatapoints);
}
markAsTouched() {
if (!this.touched) {
this.onTouched();
this.touched = true;
}
}
includesSearchString(datapoint, lowerCaseSearchString) {
const label = datapoint.label?.toLowerCase();
if (label && label.includes(lowerCaseSearchString)) {
return true;
}
const fragment = datapoint.fragment?.toLowerCase();
if (fragment && fragment.includes(lowerCaseSearchString)) {
return true;
}
const series = datapoint.series?.toLowerCase();
if (series && series.includes(lowerCaseSearchString)) {
return true;
}
return false;
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DatapointSelectorComponent, deps: [{ token: i1.DatapointLibraryService }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DatapointSelectorComponent, selector: "c8y-datapoint-selector", inputs: { contextAsset: "contextAsset", allowChangingContext: "allowChangingContext", allowDatapointsFromMultipleAssets: "allowDatapointsFromMultipleAssets", defaultActiveState: "defaultActiveState", ignoreDatapointTemplates: "ignoreDatapointTemplates", datapointTemplatesOnly: "datapointTemplatesOnly", guessDatapointUnit: "guessDatapointUnit", allowSearch: "allowSearch", hideSelection: "hideSelection", itemsEditable: "itemsEditable" }, providers: [
{
provide: NG_VALUE_ACCESSOR,
multi: true,
useExisting: forwardRef(() => DatapointSelectorComponent)
}
], ngImport: i0, template: "<div\n class=\"d-grid grid__row--1 fit-h\"\n [ngClass]=\"{\n 'grid__col--3-6-3--md': allowChangingContext && !hideSelection,\n 'grid__col--8-4--md': !allowChangingContext && !hideSelection,\n 'grid__col--4-8--md': allowChangingContext && hideSelection\n }\"\n>\n <div class=\"d-flex d-col p-relative bg-level-1\" *ngIf=\"allowChangingContext\">\n <c8y-asset-selector-miller\n class=\"d-contents\"\n [(ngModel)]=\"contextAsset\"\n [asset]=\"contextAsset\"\n (onSelected)=\"selectionChanged($event)\"\n [container]=\"''\"\n [config]=\"{\n view: 'miller',\n groupsSelectable: true,\n columnHeaders: true,\n showChildDevices: true,\n showUnassignedDevices: true,\n singleColumn: true,\n search: allowSearch,\n showFilter: true\n }\"\n ></c8y-asset-selector-miller>\n </div>\n <!-- center column -->\n <div class=\"inner-scroll bg-component\">\n <ng-template #noDeviceEmptyState>\n <div class=\"p-16\">\n <c8y-ui-empty-state\n [icon]=\"'c8y-data-points'\"\n [title]=\"emptyStateTitle | translate\"\n [subtitle]=\"'Select an asset from the list.' | translate\"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </div>\n </ng-template>\n <ng-template #loadingData>\n <div class=\"p-16 text-center\">\n <c8y-loading></c8y-loading>\n </div>\n </ng-template>\n <div class=\"bg-inherit\" *ngIf=\"assetSelection | async as asset; else noDeviceEmptyState\">\n <div class=\"p-l-16 p-r-16 p-t-8 p-b-8 sticky-top bg-inherit separator-bottom\">\n <p\n class=\"text-medium text-truncate\"\n [title]=\"selectorTitle | translate\"\n >\n {{ selectorTitle | translate }}\n </p>\n <div class=\"input-group input-group-search m-t-4\" id=\"search\" *ngIf=\"!loadingDatapoints\">\n <input\n class=\"form-control\"\n type=\"search\"\n placeholder=\"Search\u2026\"\n [ngModel]=\"searchString\"\n (ngModelChange)=\"searchStringChanged($event)\"\n />\n <span class=\"input-group-addon\">\n <i c8yIcon=\"search\" *ngIf=\"!searchString; else clearSearchString\"></i>\n <ng-template #clearSearchString>\n <i class=\"text-muted\" c8yIcon=\"times\" (click)=\"searchStringChanged()\"></i>\n </ng-template>\n </span>\n </div>\n </div>\n <ng-container *ngIf=\"filteredDatapoints$ | async as filteredDatapoints; else loadingData\">\n <ng-container *ngIf=\"!loadingDatapoints; else loadingData\">\n <ng-container *ngIf=\"datapoints$ | async as datapoints\">\n <div class=\"p-16\" *ngIf=\"!filteredDatapoints.length\">\n <c8y-ui-empty-state\n [icon]=\"'c8y-data-points'\"\n [title]=\"emptyStateTitle | translate\"\n [subtitle]=\"\n datapoints.length\n ? (emptyStateSubtitleWhenNoMatchingDataPoints | translate)\n : (emptyStateSubtitleWhenNoDataPointsInAsset | translate)\n \"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </div>\n\n <c8y-list-group>\n <c8y-list-item\n class=\"sticky-top\"\n style=\"top: 72px\"\n *ngIf=\"\n datapoints.length > maxNumberOfDatapoints &&\n filteredDatapoints.length >= maxNumberOfDatapoints\n \"\n >\n <div class=\"alert alert-warning m-b-0\">\n {{\n 'Due to the large number, only a subset of data points is displayed. Use search to narrow down the number of results.'\n | translate\n }}\n </div>\n </c8y-list-item>\n <c8y-datapoint-selector-list-item\n class=\"d-contents\"\n [ngModel]=\"dp\"\n [isSelected]=\"selectedDatapoints | includesDatapoint: dp\"\n [datapointLibraryEntries]=\"datapointLibraryEntries\"\n [disableTypeaheadIfSelected]=\"true\"\n [addButtonType]=\"hideSelection ? AddButtonTypes.select : AddButtonTypes.addRemove\"\n (added)=\"datapointAdded($event)\"\n (removed)=\"datapointRemoved($event)\"\n [highlightText]=\"searchStringChanges$ | async\"\n *ngFor=\"let dp of filteredDatapoints; trackBy: trackByFn\"\n [editable]=\"itemsEditable\"\n ></c8y-datapoint-selector-list-item>\n </c8y-list-group>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </div>\n <!-- last column -->\n <div class=\"inner-scroll bg-level-1\" *ngIf=\"!hideSelection\">\n <p class=\"text-medium p-l-16 p-r-16 p-t-8 p-b-8 separator-bottom sticky-top text-truncate\"\n [title]=\"selectedListTitle | translate\" translate>{{ selectedListTitle }}</p>\n <div\n class=\"d-flex flex-wrap gap-8 p-l-16 p-r-16 p-t-8 p-b-16\"\n *ngIf=\"selectedDatapoints?.length\"\n >\n <div class=\"c8y-datapoint-pill\" *ngFor=\"let selectedDp of selectedDatapoints\">\n <button\n class=\"c8y-datapoint-pill__btn\"\n type=\"button\"\n [title]=\"'Remove' | translate\"\n (click)=\"datapointRemoved(selectedDp)\"\n >\n <i class=\"icon-14\" c8yIcon=\"remove\"></i>\n </button>\n <div\n class=\"c8y-datapoint-pill__label\"\n [title]=\"selectedDp | datapointLabel: { doNotUseLabel: true, includeDevice: true }\"\n >\n <i class=\"m-r-4 icon-14\" c8yIcon=\"circle\" [style.color]=\"selectedDp.color\"></i>\n <span class=\"text-truncate\">\n <span class=\"text-truncate\">{{ selectedDp | datapointLabel }}</span>\n <small class=\"text-muted text-10\" *ngIf=\"selectedDp?.__target?.name\">\n {{ selectedDp?.__target?.name }}\n </small>\n </span>\n </div>\n </div>\n </div>\n <div class=\"p-r-8\" *ngIf=\"!selectedDatapoints || !selectedDatapoints.length\">\n <c8y-ui-empty-state\n [icon]=\"'c8y-data-points'\"\n [title]=\"emptyStateTitle | translate\"\n [subtitle]=\"\n 'Select the asset, then on the available data points list, click on the plus button on the desired data point.'\n | translate\n \"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </div>\n </div>\n</div>\n", dependencies: [{ kind: "component", type: i2.MillerViewComponent, selector: "c8y-asset-selector-miller", inputs: ["config", "asset", "selectedDevice", "rootNode", "container"], outputs: ["onSelected", "onClearSelected"] }, { kind: "component", type: i3.EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "directive", type: i3.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i3.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.LoadingComponent, selector: "c8y-loading", inputs: ["layout", "progress", "message"] }, { kind: "directive", type: i5.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i3.ListGroupComponent, selector: "c8y-list-group" }, { kind: "component", type: i3.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i6.DatapointSelectorListItemComponent, selector: "c8y-datapoint-selector-list-item", inputs: ["defaultFormOptions", "isSelected", "isCollapsed", "addButtonType", "editable", "showActiveToggle", "activeToggleDisabled", "showOptions", "datapointLibraryEntries", "actions", "optionToRemove", "hasUnlinkTemplateOption", "colorPickerDisabled", "disableTypeaheadIfSelected", "highlightText"], outputs: ["added", "removed"] }, { kind: "pipe", type: i3.C8yTranslatePipe, name: "translate" }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i7.IncludesDatapointPipe, name: "includesDatapoint" }, { kind: "pipe", type: i8.DatapointLabelPipe, name: "datapointLabel" }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DatapointSelectorComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-datapoint-selector', providers: [
{
provide: NG_VALUE_ACCESSOR,
multi: true,
useExisting: forwardRef(() => DatapointSelectorComponent)
}
], template: "<div\n class=\"d-grid grid__row--1 fit-h\"\n [ngClass]=\"{\n 'grid__col--3-6-3--md': allowChangingContext && !hideSelection,\n 'grid__col--8-4--md': !allowChangingContext && !hideSelection,\n 'grid__col--4-8--md': allowChangingContext && hideSelection\n }\"\n>\n <div class=\"d-flex d-col p-relative bg-level-1\" *ngIf=\"allowChangingContext\">\n <c8y-asset-selector-miller\n class=\"d-contents\"\n [(ngModel)]=\"contextAsset\"\n [asset]=\"contextAsset\"\n (onSelected)=\"selectionChanged($event)\"\n [container]=\"''\"\n [config]=\"{\n view: 'miller',\n groupsSelectable: true,\n columnHeaders: true,\n showChildDevices: true,\n showUnassignedDevices: true,\n singleColumn: true,\n search: allowSearch,\n showFilter: true\n }\"\n ></c8y-asset-selector-miller>\n </div>\n <!-- center column -->\n <div class=\"inner-scroll bg-component\">\n <ng-template #noDeviceEmptyState>\n <div class=\"p-16\">\n <c8y-ui-empty-state\n [icon]=\"'c8y-data-points'\"\n [title]=\"emptyStateTitle | translate\"\n [subtitle]=\"'Select an asset from the list.' | translate\"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </div>\n </ng-template>\n <ng-template #loadingData>\n <div class=\"p-16 text-center\">\n <c8y-loading></c8y-loading>\n </div>\n </ng-template>\n <div class=\"bg-inherit\" *ngIf=\"assetSelection | async as asset; else noDeviceEmptyState\">\n <div class=\"p-l-16 p-r-16 p-t-8 p-b-8 sticky-top bg-inherit separator-bottom\">\n <p\n class=\"text-medium text-truncate\"\n [title]=\"selectorTitle | translate\"\n >\n {{ selectorTitle | translate }}\n </p>\n <div class=\"input-group input-group-search m-t-4\" id=\"search\" *ngIf=\"!loadingDatapoints\">\n <input\n class=\"form-control\"\n type=\"search\"\n placeholder=\"Search\u2026\"\n [ngModel]=\"searchString\"\n (ngModelChange)=\"searchStringChanged($event)\"\n />\n <span class=\"input-group-addon\">\n <i c8yIcon=\"search\" *ngIf=\"!searchString; else clearSearchString\"></i>\n <ng-template #clearSearchString>\n <i class=\"text-muted\" c8yIcon=\"times\" (click)=\"searchStringChanged()\"></i>\n </ng-template>\n </span>\n </div>\n </div>\n <ng-container *ngIf=\"filteredDatapoints$ | async as filteredDatapoints; else loadingData\">\n <ng-container *ngIf=\"!loadingDatapoints; else loadingData\">\n <ng-container *ngIf=\"datapoints$ | async as datapoints\">\n <div class=\"p-16\" *ngIf=\"!filteredDatapoints.length\">\n <c8y-ui-empty-state\n [icon]=\"'c8y-data-points'\"\n [title]=\"emptyStateTitle | translate\"\n [subtitle]=\"\n datapoints.length\n ? (emptyStateSubtitleWhenNoMatchingDataPoints | translate)\n : (emptyStateSubtitleWhenNoDataPointsInAsset | translate)\n \"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </div>\n\n <c8y-list-group>\n <c8y-list-item\n class=\"sticky-top\"\n style=\"top: 72px\"\n *ngIf=\"\n datapoints.length > maxNumberOfDatapoints &&\n filteredDatapoints.length >= maxNumberOfDatapoints\n \"\n >\n <div class=\"alert alert-warning m-b-0\">\n {{\n 'Due to the large number, only a subset of data points is displayed. Use search to narrow down the number of results.'\n | translate\n }}\n </div>\n </c8y-list-item>\n <c8y-datapoint-selector-list-item\n class=\"d-contents\"\n [ngModel]=\"dp\"\n [isSelected]=\"selectedDatapoints | includesDatapoint: dp\"\n [datapointLibraryEntries]=\"datapointLibraryEntries\"\n [disableTypeaheadIfSelected]=\"true\"\n [addButtonType]=\"hideSelection ? AddButtonTypes.select : AddButtonTypes.addRemove\"\n (added)=\"datapointAdded($event)\"\n (removed)=\"datapointRemoved($event)\"\n [highlightText]=\"searchStringChanges$ | async\"\n *ngFor=\"let dp of filteredDatapoints; trackBy: trackByFn\"\n [editable]=\"itemsEditable\"\n ></c8y-datapoint-selector-list-item>\n </c8y-list-group>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </div>\n <!-- last column -->\n <div class=\"inner-scroll bg-level-1\" *ngIf=\"!hideSelection\">\n <p class=\"text-medium p-l-16 p-r-16 p-t-8 p-b-8 separator-bottom sticky-top text-truncate\"\n [title]=\"selectedListTitle | translate\" translate>{{ selectedListTitle }}</p>\n <div\n class=\"d-flex flex-wrap gap-8 p-l-16 p-r-16 p-t-8 p-b-16\"\n *ngIf=\"selectedDatapoints?.length\"\n >\n <div class=\"c8y-datapoint-pill\" *ngFor=\"let selectedDp of selectedDatapoints\">\n <button\n class=\"c8y-datapoint-pill__btn\"\n type=\"button\"\n [title]=\"'Remove' | translate\"\n (click)=\"datapointRemoved(selectedDp)\"\n >\n <i class=\"icon-14\" c8yIcon=\"remove\"></i>\n </button>\n <div\n class=\"c8y-datapoint-pill__label\"\n [title]=\"selectedDp | datapointLabel: { doNotUseLabel: true, includeDevice: true }\"\n >\n <i class=\"m-r-4 icon-14\" c8yIcon=\"circle\" [style.color]=\"selectedDp.color\"></i>\n <span class=\"text-truncate\">\n <span class=\"text-truncate\">{{ selectedDp | datapointLabel }}</span>\n <small class=\"text-muted text-10\" *ngIf=\"selectedDp?.__target?.name\">\n {{ selectedDp?.__target?.name }}\n </small>\n </span>\n </div>\n </div>\n </div>\n <div class=\"p-r-8\" *ngIf=\"!selectedDatapoints || !selectedDatapoints.length\">\n <c8y-ui-empty-state\n [icon]=\"'c8y-data-points'\"\n [title]=\"emptyStateTitle | translate\"\n [subtitle]=\"\n 'Select the asset, then on the available data points list, click on the plus button on the desired data point.'\n | translate\n \"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </div>\n </div>\n</div>\n" }]
}], ctorParameters: () => [{ type: i1.DatapointLibraryService }], propDecorators: { contextAsset: [{
type: Input
}], allowChangingContext: [{
type: Input
}], allowDatapointsFromMultipleAssets: [{
type: Input
}], defaultActiveState: [{
type: Input
}], ignoreDatapointTemplates: [{
type: Input
}], datapointTemplatesOnly: [{
type: Input
}], guessDatapointUnit: [{
type: Input
}], allowSearch: [{
type: Input
}], hideSelection: [{
type: Input
}], itemsEditable: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datapoint-selector.component.js","sourceRoot":"","sources":["../../../datapoint-selector/datapoint-selector.component.ts","../../../datapoint-selector/datapoint-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, forwardRef, Input, OnInit } from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { IIdentified, IResultList } from '@c8y/client';\nimport { gettext } from '@c8y/ngx-components';\nimport { BehaviorSubject, combineLatest, from, Observable } from 'rxjs';\nimport {\n  debounceTime,\n  distinctUntilChanged,\n  map,\n  shareReplay,\n  switchMap,\n  tap\n} from 'rxjs/operators';\nimport { DatapointLibraryService } from './datapoint-library.service';\nimport { KPIDetails, ManagedObjectKPI } from './datapoint-selection.model';\nimport { AddButtonTypes } from './datapoint-selector-list-item/datapoint-selector-list-item.component';\n\n@Component({\n  selector: 'c8y-datapoint-selector',\n  templateUrl: './datapoint-selector.component.html',\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      multi: true,\n      useExisting: forwardRef(() => DatapointSelectorComponent)\n    }\n  ]\n})\nexport class DatapointSelectorComponent implements OnInit, ControlValueAccessor {\n  @Input() contextAsset: IIdentified;\n  @Input() allowChangingContext = true;\n  @Input() allowDatapointsFromMultipleAssets = true;\n  selectedDatapoints = new Array<KPIDetails>();\n  @Input() defaultActiveState = true;\n  @Input() ignoreDatapointTemplates = false;\n  @Input() datapointTemplatesOnly = false;\n  @Input() guessDatapointUnit = true;\n  @Input() allowSearch = true;\n  @Input() hideSelection = false;\n  @Input() itemsEditable = true;\n  searchString = '';\n  maxNumberOfDatapoints = 50;\n  selectorTitle: string;\n  emptyStateTitle: string;\n  selectedListTitle: string;\n  AddButtonTypes = AddButtonTypes;\n\n  loadingDatapoints = false;\n  assetSelection = new BehaviorSubject<IIdentified>(null);\n  datapoints$: Observable<KPIDetails[]>;\n  filteredDatapoints$: Observable<KPIDetails[]>;\n  searchStringChanges$: Observable<string>;\n  datapointLibraryEntries: Observable<IResultList<ManagedObjectKPI>>;\n\n  emptyStateSubtitleWhenNoMatchingDataPoints = gettext('Try another search term.');\n  emptyStateSubtitleWhenNoDataPointsInAsset = gettext(\n    'Select an asset with data points from the list.'\n  );\n\n  private searchString$ = new BehaviorSubject('');\n  private touched = false;\n  private onChange: (quantity: KPIDetails[]) => void;\n  private onTouched: () => void;\n\n  constructor(private datapointService: DatapointLibraryService) {}\n\n  ngOnInit(): void {\n    this.selectorTitle = this.datapointTemplatesOnly\n      ? gettext('Available data point templates')\n      : gettext('Available data points');\n    this.emptyStateTitle = this.datapointTemplatesOnly\n      ? gettext('No data point templates to display.')\n      : gettext('No data points to display.');\n    this.selectedListTitle = this.datapointTemplatesOnly\n      ? gettext('Selected data point templates')\n      : gettext('Selected data points');\n    this.setupObservables();\n    if (!this.ignoreDatapointTemplates) {\n      this.datapointLibraryEntries = from(this.datapointService.getFirstDatapointLibraryPage());\n    }\n\n    if (this.contextAsset) {\n      this.selectionChanged(this.contextAsset);\n    }\n  }\n\n  writeValue(obj: KPIDetails[]): void {\n    this.selectedDatapoints = obj;\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  datapointAdded(dp: KPIDetails): void {\n    this.markAsTouched();\n    dp.__active = this.defaultActiveState;\n    if (this.guessDatapointUnit && !dp.unit) {\n      this.datapointService.guessUnitOfDatapoint(dp.fragment, dp.series, dp.__target).then(unit => {\n        dp.unit = unit;\n      });\n    }\n    this.selectedDatapoints = [...this.selectedDatapoints, dp];\n    this.emitCurrentSelection();\n  }\n\n  datapointRemoved(dp: KPIDetails): void {\n    this.markAsTouched();\n    this.selectedDatapoints = this.selectedDatapoints.filter(\n      tmp =>\n        tmp.fragment !== dp.fragment ||\n        tmp.series !== dp.series ||\n        tmp.__target?.id !== dp.__target?.id\n    );\n    this.emitCurrentSelection();\n  }\n\n  selectionChanged(evt: IIdentified | IIdentified[]): void {\n    if (Array.isArray(evt) && evt.length !== 0) {\n      return this.selectAsset(evt[0]);\n    }\n\n    if (!Array.isArray(evt) && evt.items) {\n      return this.selectionChanged(evt.items);\n    }\n\n    if (!Array.isArray(evt) && evt.id) {\n      return this.selectAsset(evt);\n    }\n\n    // reset selection\n    this.assetSelection.next(null);\n  }\n\n  trackByFn(_index: number, item: KPIDetails): string {\n    return `${item.fragment}-${item.__target?.id}-${item.series}`;\n  }\n\n  searchStringChanged(newValue = ''): void {\n    this.searchString$.next(newValue);\n    this.searchString = newValue;\n  }\n\n  private setupObservables(): void {\n    this.datapoints$ = this.assetSelection.pipe(\n      tap(() => {\n        this.loadingDatapoints = true;\n      }),\n      switchMap(asset =>\n        asset?.id\n          ? this.datapointService.getDatapointsOfAsset(\n              asset,\n              this.ignoreDatapointTemplates,\n              this.datapointTemplatesOnly\n            )\n          : []\n      ),\n      tap(() => (this.loadingDatapoints = false)),\n      shareReplay(1)\n    );\n\n    this.searchStringChanges$ = this.searchString$.pipe(\n      distinctUntilChanged(),\n      debounceTime(500),\n      shareReplay(1)\n    );\n\n    this.filteredDatapoints$ = combineLatest([this.searchStringChanges$, this.datapoints$]).pipe(\n      map(([searchString, datapoints]) => {\n        if (!searchString) {\n          return datapoints;\n        }\n        const lowerCaseSearchString = searchString.toLowerCase();\n        return datapoints.filter(datapoint =>\n          this.includesSearchString(datapoint, lowerCaseSearchString)\n        );\n      }),\n      map(filtered => filtered.slice(0, this.maxNumberOfDatapoints))\n    );\n  }\n\n  private selectAsset(asset: IIdentified) {\n    this.assetSelection.next(asset);\n    this.searchStringChanged();\n    if (!this.allowDatapointsFromMultipleAssets) {\n      this.clearSelection();\n    }\n  }\n\n  private clearSelection(): void {\n    this.selectedDatapoints = [];\n    this.emitCurrentSelection();\n  }\n\n  private emitCurrentSelection() {\n    this.onChange(this.selectedDatapoints);\n  }\n\n  private markAsTouched() {\n    if (!this.touched) {\n      this.onTouched();\n      this.touched = true;\n    }\n  }\n\n  private includesSearchString(datapoint: KPIDetails, lowerCaseSearchString: string): boolean {\n    const label = datapoint.label?.toLowerCase();\n    if (label && label.includes(lowerCaseSearchString)) {\n      return true;\n    }\n\n    const fragment = datapoint.fragment?.toLowerCase();\n    if (fragment && fragment.includes(lowerCaseSearchString)) {\n      return true;\n    }\n\n    const series = datapoint.series?.toLowerCase();\n    if (series && series.includes(lowerCaseSearchString)) {\n      return true;\n    }\n\n    return false;\n  }\n}\n","<div\n  class=\"d-grid grid__row--1 fit-h\"\n  [ngClass]=\"{\n    'grid__col--3-6-3--md': allowChangingContext && !hideSelection,\n    'grid__col--8-4--md': !allowChangingContext && !hideSelection,\n    'grid__col--4-8--md': allowChangingContext && hideSelection\n  }\"\n>\n  <div class=\"d-flex d-col p-relative bg-level-1\" *ngIf=\"allowChangingContext\">\n    <c8y-asset-selector-miller\n      class=\"d-contents\"\n      [(ngModel)]=\"contextAsset\"\n      [asset]=\"contextAsset\"\n      (onSelected)=\"selectionChanged($event)\"\n      [container]=\"''\"\n      [config]=\"{\n        view: 'miller',\n        groupsSelectable: true,\n        columnHeaders: true,\n        showChildDevices: true,\n        showUnassignedDevices: true,\n        singleColumn: true,\n        search: allowSearch,\n        showFilter: true\n      }\"\n    ></c8y-asset-selector-miller>\n  </div>\n  <!-- center column -->\n  <div class=\"inner-scroll bg-component\">\n    <ng-template #noDeviceEmptyState>\n      <div class=\"p-16\">\n        <c8y-ui-empty-state\n          [icon]=\"'c8y-data-points'\"\n          [title]=\"emptyStateTitle | translate\"\n          [subtitle]=\"'Select an asset from the list.' | translate\"\n          [horizontal]=\"true\"\n        ></c8y-ui-empty-state>\n      </div>\n    </ng-template>\n    <ng-template #loadingData>\n      <div class=\"p-16 text-center\">\n        <c8y-loading></c8y-loading>\n      </div>\n    </ng-template>\n    <div class=\"bg-inherit\" *ngIf=\"assetSelection | async as asset; else noDeviceEmptyState\">\n      <div class=\"p-l-16 p-r-16 p-t-8 p-b-8 sticky-top bg-inherit separator-bottom\">\n        <p\n          class=\"text-medium text-truncate\"\n          [title]=\"selectorTitle | translate\"\n        >\n          {{ selectorTitle | translate }}\n        </p>\n        <div class=\"input-group input-group-search m-t-4\" id=\"search\" *ngIf=\"!loadingDatapoints\">\n          <input\n            class=\"form-control\"\n            type=\"search\"\n            placeholder=\"Search…\"\n            [ngModel]=\"searchString\"\n            (ngModelChange)=\"searchStringChanged($event)\"\n          />\n          <span class=\"input-group-addon\">\n            <i c8yIcon=\"search\" *ngIf=\"!searchString; else clearSearchString\"></i>\n            <ng-template #clearSearchString>\n              <i class=\"text-muted\" c8yIcon=\"times\" (click)=\"searchStringChanged()\"></i>\n            </ng-template>\n          </span>\n        </div>\n      </div>\n      <ng-container *ngIf=\"filteredDatapoints$ | async as filteredDatapoints; else loadingData\">\n        <ng-container *ngIf=\"!loadingDatapoints; else loadingData\">\n          <ng-container *ngIf=\"datapoints$ | async as datapoints\">\n            <div class=\"p-16\" *ngIf=\"!filteredDatapoints.length\">\n              <c8y-ui-empty-state\n                [icon]=\"'c8y-data-points'\"\n                [title]=\"emptyStateTitle | translate\"\n                [subtitle]=\"\n                  datapoints.length\n                    ? (emptyStateSubtitleWhenNoMatchingDataPoints | translate)\n                    : (emptyStateSubtitleWhenNoDataPointsInAsset | translate)\n                \"\n                [horizontal]=\"true\"\n              ></c8y-ui-empty-state>\n            </div>\n\n            <c8y-list-group>\n              <c8y-list-item\n                class=\"sticky-top\"\n                style=\"top: 72px\"\n                *ngIf=\"\n                  datapoints.length > maxNumberOfDatapoints &&\n                  filteredDatapoints.length >= maxNumberOfDatapoints\n                \"\n              >\n                <div class=\"alert alert-warning m-b-0\">\n                  {{\n                    'Due to the large number, only a subset of data points is displayed. Use search to narrow down the number of results.'\n                      | translate\n                  }}\n                </div>\n              </c8y-list-item>\n              <c8y-datapoint-selector-list-item\n                class=\"d-contents\"\n                [ngModel]=\"dp\"\n                [isSelected]=\"selectedDatapoints | includesDatapoint: dp\"\n                [datapointLibraryEntries]=\"datapointLibraryEntries\"\n                [disableTypeaheadIfSelected]=\"true\"\n                [addButtonType]=\"hideSelection ? AddButtonTypes.select : AddButtonTypes.addRemove\"\n                (added)=\"datapointAdded($event)\"\n                (removed)=\"datapointRemoved($event)\"\n                [highlightText]=\"searchStringChanges$ | async\"\n                *ngFor=\"let dp of filteredDatapoints; trackBy: trackByFn\"\n                [editable]=\"itemsEditable\"\n              ></c8y-datapoint-selector-list-item>\n            </c8y-list-group>\n          </ng-container>\n        </ng-container>\n      </ng-container>\n    </div>\n  </div>\n  <!-- last column  -->\n  <div class=\"inner-scroll bg-level-1\" *ngIf=\"!hideSelection\">\n    <p class=\"text-medium p-l-16 p-r-16 p-t-8 p-b-8 separator-bottom sticky-top text-truncate\"\n      [title]=\"selectedListTitle | translate\" translate>{{ selectedListTitle }}</p>\n    <div\n      class=\"d-flex flex-wrap gap-8 p-l-16 p-r-16 p-t-8 p-b-16\"\n      *ngIf=\"selectedDatapoints?.length\"\n    >\n      <div clas