@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
175 lines • 36.1 kB
JavaScript
import { moveItemInArray } from '@angular/cdk/drag-drop';
import { Component, forwardRef, Input, Optional, Output } from '@angular/core';
import { FormBuilder, NG_VALIDATORS, NG_VALUE_ACCESSOR } from '@angular/forms';
import { C8yValidators } from '@c8y/ngx-components';
import { WidgetConfigComponent } from '@c8y/ngx-components/context-dashboard';
import { from, Observable } from 'rxjs';
import { map, shareReplay, take, tap } from 'rxjs/operators';
import { DatapointLibraryService } from '../datapoint-library.service';
import { DatapointSelectorService } from '../datapoint-selector.service';
import { AddButtonTypes } from '../datapoint-selector-list-item/datapoint-selector-list-item.component';
import * as i0 from "@angular/core";
import * as i1 from "../datapoint-selector.service";
import * as i2 from "../datapoint-library.service";
import * as i3 from "@angular/forms";
import * as i4 from "@c8y/ngx-components/context-dashboard";
import * as i5 from "@c8y/ngx-components";
import * as i6 from "@angular/common";
import * as i7 from "@angular/cdk/drag-drop";
import * as i8 from "../datapoint-selector-list-item/datapoint-selector-list-item.component";
export class DatapointSelectionListComponent {
constructor(datapointSelector, datapointLibrary, formBuilder, widgetComponent) {
this.datapointSelector = datapointSelector;
this.datapointLibrary = datapointLibrary;
this.formBuilder = formBuilder;
this.widgetComponent = widgetComponent;
this.actions = [];
this.allowDragAndDrop = true;
this.config = {};
this.defaultFormOptions = {};
this.minActiveCount = 1;
this.resolveContext = true;
this.listTitle = '';
this.maxActiveCountReached = false;
this.AddButtonTypes = AddButtonTypes;
this.usedValidators = {};
this.formArray = this.formBuilder.array([]);
this.isValid = this.formArray.statusChanges.pipe(map(status => status === 'VALID'));
this.datapointLibraryEntries = from(this.datapointLibrary.getFirstDatapointLibraryPage()).pipe(shareReplay());
this.change = this.formArray.valueChanges.pipe(map(res => this.transformValue(res)));
}
ngOnChanges(changes) {
if (!changes.maxActiveCount && !changes.minActiveCount) {
return;
}
if (changes.maxActiveCount) {
this.usedValidators.maxActiveCount = C8yValidators.maxActiveCount(this.maxActiveCount);
}
if (changes.minActiveCount) {
this.usedValidators.minActiveCount = C8yValidators.minActiveCount(this.minActiveCount);
}
const validators = Object.values(this.usedValidators);
this.formArray.setValidators(validators);
}
registerOnTouched(fn) {
this.formArray.valueChanges.pipe(take(1)).subscribe(fn);
}
validate(_control) {
return this.formArray.valid ? null : { formInvalid: {} };
}
ngOnInit() {
const context = this.widgetComponent?.context;
if (context?.id && this.resolveContext) {
const { name, id, c8y_IsDevice } = context;
this.config.contextAsset = { name, id, c8y_IsDevice };
}
}
writeValue(obj) {
this.formArray.clear();
if (obj?.length) {
obj.forEach(val => {
const formgroup = this.formBuilder.group({ details: [] });
formgroup.patchValue({ details: val });
this.formArray.push(formgroup);
});
}
this.calculateMaxActiveCount();
}
registerOnChange(fn) {
this.formArray.valueChanges
.pipe(map(res => this.transformValue(res)),
// check maxActiveCount
tap(() => {
this.calculateMaxActiveCount();
}))
.subscribe(fn);
}
add() {
const allowChangingContext = !this.widgetComponent?.isDeviceTypeDashboard && this.config?.allowChangingContext !== false;
this.datapointSelector
.selectDataPoints({
...(this.config || {}),
selectedDatapoints: this.transformValue(this.formArray.value),
defaultActiveState: true,
allowChangingContext,
allowSearch: !this.config?.contextAsset
})
.then(result => {
this.writeValue(result);
}, () => {
// nothing to do, modal was closed
});
}
onItemRemoved(index) {
this.formArray.removeAt(index);
}
drop(event) {
const currentSorting = this.formArray.value;
moveItemInArray(currentSorting, event.previousIndex, event.currentIndex);
this.formArray.setValue(currentSorting);
}
transformValue(formArrayValue) {
if (!formArrayValue) {
return [];
}
return formArrayValue.map(tmp => Object.assign({}, ...Object.values(tmp)));
}
calculateMaxActiveCount() {
if (this.maxActiveCount) {
const currentlyActive = this.formArray.value.filter(tmp => tmp.details?.__active).length;
this.maxActiveCountReached = currentlyActive >= this.maxActiveCount;
}
this.maxActiveCountReached = false;
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DatapointSelectionListComponent, deps: [{ token: i1.DatapointSelectorService }, { token: i2.DatapointLibraryService }, { token: i3.FormBuilder }, { token: i4.WidgetConfigComponent, optional: true }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DatapointSelectionListComponent, selector: "c8y-datapoint-selection-list", inputs: { actions: "actions", allowDragAndDrop: "allowDragAndDrop", config: "config", defaultFormOptions: "defaultFormOptions", maxActiveCount: "maxActiveCount", minActiveCount: "minActiveCount", resolveContext: "resolveContext", listTitle: "listTitle" }, outputs: { isValid: "isValid", change: "change" }, providers: [
{
provide: NG_VALUE_ACCESSOR,
multi: true,
useExisting: forwardRef(() => DatapointSelectionListComponent)
},
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => DatapointSelectionListComponent),
multi: true
}
], usesOnChanges: true, ngImport: i0, template: "<div class=\"card-header separator sticky-top bg-inherit\">\n <span\n class=\"card-title h4\"\n *ngIf=\"listTitle\"\n >\n {{ listTitle | translate }}\n </span>\n <span\n class=\"card-title h4\"\n *ngIf=\"!listTitle\"\n >\n {{ 'Data points' | translate }}\n </span>\n</div>\n\n<c8y-list-group\n class=\"flex-grow ff-scroll-fix cdk-droplist\"\n cdkDropList\n (cdkDropListDropped)=\"drop($event)\"\n [cdkDropListDisabled]=\"!allowDragAndDrop || formArray.controls?.length < 2\"\n>\n <div\n class=\"alert alert-warning m-t-8\"\n role=\"alert\"\n ngNonBindable\n *ngIf=\"formArray.errors?.minActiveCount\"\n translate\n [translateParams]=\"formArray.errors?.minActiveCount\"\n >\n At least {{ minActive }} active data points must be selected.\n </div>\n\n <div\n class=\"alert alert-warning m-t-8\"\n role=\"alert\"\n ngNonBindable\n *ngIf=\"formArray.errors?.maxActiveCount\"\n translate\n [translateParams]=\"formArray.errors?.maxActiveCount\"\n >\n At maximum {{ maxActive }} active data points are allowed to be selected.\n </div>\n\n <ng-content select=\".alert\"></ng-content>\n\n <div\n class=\"p-t-8\"\n *ngIf=\"!formArray.controls?.length\"\n >\n <c8y-ui-empty-state\n class=\"p-t-8\"\n [icon]=\"'c8y-data-points'\"\n [title]=\"'No data points to display.' | translate\"\n [subtitle]=\"'Add your first data point.' | translate\"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </div>\n <div\n [formGroup]=\"dpForm\"\n *ngFor=\"let dpForm of formArray.controls; let index = index\"\n >\n <c8y-datapoint-selector-list-item\n class=\"d-block\"\n [defaultFormOptions]=\"defaultFormOptions\"\n [activeToggleDisabled]=\"maxActiveCountReached\"\n [showActiveToggle]=\"true\"\n [addButtonType]=\"AddButtonTypes.none\"\n [showOptions]=\"true\"\n [editable]=\"true\"\n [colorPickerDisabled]=\"false\"\n [actions]=\"actions\"\n [optionToRemove]=\"true\"\n [datapointLibraryEntries]=\"datapointLibraryEntries\"\n [hasUnlinkTemplateOption]=\"true\"\n formControlName=\"details\"\n (removed)=\"onItemRemoved(index)\"\n cdkDrag\n >\n <c8y-li-drag-handle\n title=\"{{ 'Click and drag to reorder' | translate }}\"\n cdkDragHandle\n >\n <i c8yIcon=\"drag-reorder\"></i>\n </c8y-li-drag-handle>\n </c8y-datapoint-selector-list-item>\n </div>\n</c8y-list-group>\n\n<div class=\"card-footer bg-inherit\">\n <button\n class=\"btn btn-default btn-sm\"\n [title]=\"'Add data point' | translate\"\n type=\"button\"\n data-cy=\"c8y-datapoint-selection-list--add-datapoint-button\"\n (click)=\"add()\"\n >\n <i c8yIcon=\"plus-circle\"></i>\n {{ 'Add data point' | translate }}\n </button>\n</div>\n", dependencies: [{ kind: "component", type: i5.EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "directive", type: i5.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i5.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "component", type: i5.ListGroupComponent, selector: "c8y-list-group" }, { kind: "component", type: i5.ListItemDragHandleComponent, selector: "c8y-list-item-drag-handle, c8y-li-drag-handle" }, { kind: "directive", type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i3.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i7.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "cdkDropListData", "cdkDropListOrientation", "id", "cdkDropListLockAxis", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListAutoScrollDisabled", "cdkDropListAutoScrollStep", "cdkDropListElementContainer"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { kind: "directive", type: i7.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragData", "cdkDragLockAxis", "cdkDragRootElement", "cdkDragBoundary", "cdkDragStartDelay", "cdkDragFreeDragPosition", "cdkDragDisabled", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragPreviewContainer", "cdkDragScale"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }, { kind: "directive", type: i7.CdkDragHandle, selector: "[cdkDragHandle]", inputs: ["cdkDragHandleDisabled"] }, { kind: "component", type: i8.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: i5.C8yTranslatePipe, name: "translate" }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DatapointSelectionListComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-datapoint-selection-list', providers: [
{
provide: NG_VALUE_ACCESSOR,
multi: true,
useExisting: forwardRef(() => DatapointSelectionListComponent)
},
{
provide: NG_VALIDATORS,
useExisting: forwardRef(() => DatapointSelectionListComponent),
multi: true
}
], template: "<div class=\"card-header separator sticky-top bg-inherit\">\n <span\n class=\"card-title h4\"\n *ngIf=\"listTitle\"\n >\n {{ listTitle | translate }}\n </span>\n <span\n class=\"card-title h4\"\n *ngIf=\"!listTitle\"\n >\n {{ 'Data points' | translate }}\n </span>\n</div>\n\n<c8y-list-group\n class=\"flex-grow ff-scroll-fix cdk-droplist\"\n cdkDropList\n (cdkDropListDropped)=\"drop($event)\"\n [cdkDropListDisabled]=\"!allowDragAndDrop || formArray.controls?.length < 2\"\n>\n <div\n class=\"alert alert-warning m-t-8\"\n role=\"alert\"\n ngNonBindable\n *ngIf=\"formArray.errors?.minActiveCount\"\n translate\n [translateParams]=\"formArray.errors?.minActiveCount\"\n >\n At least {{ minActive }} active data points must be selected.\n </div>\n\n <div\n class=\"alert alert-warning m-t-8\"\n role=\"alert\"\n ngNonBindable\n *ngIf=\"formArray.errors?.maxActiveCount\"\n translate\n [translateParams]=\"formArray.errors?.maxActiveCount\"\n >\n At maximum {{ maxActive }} active data points are allowed to be selected.\n </div>\n\n <ng-content select=\".alert\"></ng-content>\n\n <div\n class=\"p-t-8\"\n *ngIf=\"!formArray.controls?.length\"\n >\n <c8y-ui-empty-state\n class=\"p-t-8\"\n [icon]=\"'c8y-data-points'\"\n [title]=\"'No data points to display.' | translate\"\n [subtitle]=\"'Add your first data point.' | translate\"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </div>\n <div\n [formGroup]=\"dpForm\"\n *ngFor=\"let dpForm of formArray.controls; let index = index\"\n >\n <c8y-datapoint-selector-list-item\n class=\"d-block\"\n [defaultFormOptions]=\"defaultFormOptions\"\n [activeToggleDisabled]=\"maxActiveCountReached\"\n [showActiveToggle]=\"true\"\n [addButtonType]=\"AddButtonTypes.none\"\n [showOptions]=\"true\"\n [editable]=\"true\"\n [colorPickerDisabled]=\"false\"\n [actions]=\"actions\"\n [optionToRemove]=\"true\"\n [datapointLibraryEntries]=\"datapointLibraryEntries\"\n [hasUnlinkTemplateOption]=\"true\"\n formControlName=\"details\"\n (removed)=\"onItemRemoved(index)\"\n cdkDrag\n >\n <c8y-li-drag-handle\n title=\"{{ 'Click and drag to reorder' | translate }}\"\n cdkDragHandle\n >\n <i c8yIcon=\"drag-reorder\"></i>\n </c8y-li-drag-handle>\n </c8y-datapoint-selector-list-item>\n </div>\n</c8y-list-group>\n\n<div class=\"card-footer bg-inherit\">\n <button\n class=\"btn btn-default btn-sm\"\n [title]=\"'Add data point' | translate\"\n type=\"button\"\n data-cy=\"c8y-datapoint-selection-list--add-datapoint-button\"\n (click)=\"add()\"\n >\n <i c8yIcon=\"plus-circle\"></i>\n {{ 'Add data point' | translate }}\n </button>\n</div>\n" }]
}], ctorParameters: () => [{ type: i1.DatapointSelectorService }, { type: i2.DatapointLibraryService }, { type: i3.FormBuilder }, { type: i4.WidgetConfigComponent, decorators: [{
type: Optional
}] }], propDecorators: { actions: [{
type: Input
}], allowDragAndDrop: [{
type: Input
}], config: [{
type: Input
}], defaultFormOptions: [{
type: Input
}], maxActiveCount: [{
type: Input
}], minActiveCount: [{
type: Input
}], resolveContext: [{
type: Input
}], listTitle: [{
type: Input
}], isValid: [{
type: Output
}], change: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datapoint-selection-list.component.js","sourceRoot":"","sources":["../../../../datapoint-selector/datapoint-selection-list/datapoint-selection-list.component.ts","../../../../datapoint-selector/datapoint-selection-list/datapoint-selection-list.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAe,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACtE,OAAO,EACL,SAAS,EACT,UAAU,EACV,KAAK,EAGL,QAAQ,EACR,MAAM,EAEP,MAAM,eAAe,CAAC;AACvB,OAAO,EAIL,WAAW,EACX,aAAa,EACb,iBAAiB,EAIlB,MAAM,gBAAgB,CAAC;AAExB,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AACpD,OAAO,EAAE,qBAAqB,EAAE,MAAM,uCAAuC,CAAC;AAC9E,OAAO,EAAE,IAAI,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AACxC,OAAO,EAAE,GAAG,EAAE,WAAW,EAAE,IAAI,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAC7D,OAAO,EAAE,uBAAuB,EAAE,MAAM,8BAA8B,CAAC;AAQvE,OAAO,EAAE,wBAAwB,EAAE,MAAM,+BAA+B,CAAC;AACzE,OAAO,EAAE,cAAc,EAAE,MAAM,wEAAwE,CAAC;;;;;;;;;;AAkBxG,MAAM,OAAO,+BAA+B;IAoB1C,YACU,iBAA2C,EAC3C,gBAAyC,EACzC,WAAwB,EACZ,eAAsC;QAHlD,sBAAiB,GAAjB,iBAAiB,CAA0B;QAC3C,qBAAgB,GAAhB,gBAAgB,CAAyB;QACzC,gBAAW,GAAX,WAAW,CAAa;QACZ,oBAAe,GAAf,eAAe,CAAuB;QArBnD,YAAO,GAAsB,EAAE,CAAC;QAChC,qBAAgB,GAAG,IAAI,CAAC;QACxB,WAAM,GAA2C,EAAE,CAAC;QACpD,uBAAkB,GAA2C,EAAE,CAAC;QAEhE,mBAAc,GAAG,CAAC,CAAC;QACnB,mBAAc,GAAG,IAAI,CAAC;QACtB,cAAS,GAAG,EAAE,CAAC;QAGxB,0BAAqB,GAAG,KAAK,CAAC;QAC9B,mBAAc,GAAG,cAAc,CAAC;QAIxB,mBAAc,GAAmC,EAAE,CAAC;QAQ1D,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAC5C,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,KAAK,OAAO,CAAC,CAAC,CAAC;QACpF,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,4BAA4B,EAAE,CAAC,CAAC,IAAI,CAC5F,WAAW,EAAE,CACd,CAAC;QACF,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;IACvF,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,CAAC,OAAO,CAAC,cAAc,IAAI,CAAC,OAAO,CAAC,cAAc,EAAE,CAAC;YACvD,OAAO;QACT,CAAC;QACD,IAAI,OAAO,CAAC,cAAc,EAAE,CAAC;YAC3B,IAAI,CAAC,cAAc,CAAC,cAAc,GAAG,aAAa,CAAC,cAAc,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACzF,CAAC;QAED,IAAI,OAAO,CAAC,cAAc,EAAE,CAAC;YAC3B,IAAI,CAAC,cAAc,CAAC,cAAc,GAAG,aAAa,CAAC,cAAc,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACzF,CAAC;QACD,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACtD,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC;IAED,iBAAiB,CAAC,EAAO;QACvB,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;IAC1D,CAAC;IAED,QAAQ,CAAC,QAAyB;QAChC,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,WAAW,EAAE,EAAE,EAAE,CAAC;IAC3D,CAAC;IAED,QAAQ;QACN,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,EAAE,OAAO,CAAC;QAC9C,IAAI,OAAO,EAAE,EAAE,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACvC,MAAM,EAAE,IAAI,EAAE,EAAE,EAAE,YAAY,EAAE,GAAG,OAAO,CAAC;YAC3C,IAAI,CAAC,MAAM,CAAC,YAAY,GAAG,EAAE,IAAI,EAAE,EAAE,EAAE,YAAY,EAAE,CAAC;QACxD,CAAC;IACH,CAAC;IAED,UAAU,CAAC,GAAiB;QAC1B,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,CAAC;QACvB,IAAI,GAAG,EAAE,MAAM,EAAE,CAAC;YAChB,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;gBAChB,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC,CAAC;gBAC1D,SAAS,CAAC,UAAU,CAAC,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,CAAC;gBACvC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YACjC,CAAC,CAAC,CAAC;QACL,CAAC;QACD,IAAI,CAAC,uBAAuB,EAAE,CAAC;IACjC,CAAC;IAED,gBAAgB,CAAC,EAAO;QACtB,IAAI,CAAC,SAAS,CAAC,YAAY;aACxB,IAAI,CACH,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;QACpC,uBAAuB;QACvB,GAAG,CAAC,GAAG,EAAE;YACP,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACjC,CAAC,CAAC,CACH;aACA,SAAS,CAAC,EAAE,CAAC,CAAC;IACnB,CAAC;IAED,GAAG;QACD,MAAM,oBAAoB,GACxB,CAAC,IAAI,CAAC,eAAe,EAAE,qBAAqB,IAAI,IAAI,CAAC,MAAM,EAAE,oBAAoB,KAAK,KAAK,CAAC;QAC9F,IAAI,CAAC,iBAAiB;aACnB,gBAAgB,CAAC;YAChB,GAAG,CAAC,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC;YACtB,kBAAkB,EAAE,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC;YAC7D,kBAAkB,EAAE,IAAI;YACxB,oBAAoB;YACpB,WAAW,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,YAAY;SACxC,CAAC;aACD,IAAI,CACH,MAAM,CAAC,EAAE;YACP,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QAC1B,CAAC,EACD,GAAG,EAAE;YACH,kCAAkC;QACpC,CAAC,CACF,CAAC;IACN,CAAC;IAED,aAAa,CAAC,KAAa;QACzB,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED,IAAI,CAAC,KAAgC;QACnC,MAAM,cAAc,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC;QAC5C,eAAe,CAAC,cAAc,EAAE,KAAK,CAAC,aAAa,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC;QACzE,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;IAC1C,CAAC;IAEO,cAAc,CAAC,cAAqB;QAC1C,IAAI,CAAC,cAAc,EAAE,CAAC;YACpB,OAAO,EAAE,CAAC;QACZ,CAAC;QACD,OAAO,cAAc,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,GAAG,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;IAC7E,CAAC;IAEO,uBAAuB;QAC7B,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,MAAM,eAAe,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC,MAAM,CAAC;YACzF,IAAI,CAAC,qBAAqB,GAAG,eAAe,IAAI,IAAI,CAAC,cAAc,CAAC;QACtE,CAAC;QACD,IAAI,CAAC,qBAAqB,GAAG,KAAK,CAAC;IACrC,CAAC;+GArIU,+BAA+B;mGAA/B,+BAA+B,0WAb/B;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,KAAK,EAAE,IAAI;gBACX,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,+BAA+B,CAAC;aAC/D;YACD;gBACE,OAAO,EAAE,aAAa;gBACtB,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,+BAA+B,CAAC;gBAC9D,KAAK,EAAE,IAAI;aACZ;SACF,+CCpDH,kyFAoGA;;4FD9Ca,+BAA+B;kBAhB3C,SAAS;+BACE,8BAA8B,aAE7B;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,KAAK,EAAE,IAAI;4BACX,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,gCAAgC,CAAC;yBAC/D;wBACD;4BACE,OAAO,EAAE,aAAa;4BACtB,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,gCAAgC,CAAC;4BAC9D,KAAK,EAAE,IAAI;yBACZ;qBACF;;0BA0BE,QAAQ;yCArBF,OAAO;sBAAf,KAAK;gBACG,gBAAgB;sBAAxB,KAAK;gBACG,MAAM;sBAAd,KAAK;gBACG,kBAAkB;sBAA1B,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,SAAS;sBAAjB,KAAK;gBAMI,OAAO;sBAAhB,MAAM;gBACG,MAAM;sBAAf,MAAM","sourcesContent":["import { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop';\nimport {\n  Component,\n  forwardRef,\n  Input,\n  OnChanges,\n  OnInit,\n  Optional,\n  Output,\n  SimpleChanges\n} from '@angular/core';\nimport {\n  AbstractControl,\n  ControlValueAccessor,\n  FormArray,\n  FormBuilder,\n  NG_VALIDATORS,\n  NG_VALUE_ACCESSOR,\n  ValidationErrors,\n  Validator,\n  ValidatorFn\n} from '@angular/forms';\nimport { IResultList } from '@c8y/client';\nimport { C8yValidators } from '@c8y/ngx-components';\nimport { WidgetConfigComponent } from '@c8y/ngx-components/context-dashboard';\nimport { from, Observable } from 'rxjs';\nimport { map, shareReplay, take, tap } from 'rxjs/operators';\nimport { DatapointLibraryService } from '../datapoint-library.service';\nimport {\n  DatapointAction,\n  DatapointAttributesFormConfig,\n  KPIDetails,\n  ManagedObjectKPI\n} from '../datapoint-selection.model';\nimport { DatapointSelectorModalOptions } from '../datapoint-selector-modal/datapoint-selector-modal.model';\nimport { DatapointSelectorService } from '../datapoint-selector.service';\nimport { AddButtonTypes } from '../datapoint-selector-list-item/datapoint-selector-list-item.component';\n\n@Component({\n  selector: 'c8y-datapoint-selection-list',\n  templateUrl: './datapoint-selection-list.component.html',\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      multi: true,\n      useExisting: forwardRef(() => DatapointSelectionListComponent)\n    },\n    {\n      provide: NG_VALIDATORS,\n      useExisting: forwardRef(() => DatapointSelectionListComponent),\n      multi: true\n    }\n  ]\n})\nexport class DatapointSelectionListComponent\n  implements ControlValueAccessor, Validator, OnInit, OnChanges\n{\n  @Input() actions: DatapointAction[] = [];\n  @Input() allowDragAndDrop = true;\n  @Input() config: Partial<DatapointSelectorModalOptions> = {};\n  @Input() defaultFormOptions: Partial<DatapointAttributesFormConfig> = {};\n  @Input() maxActiveCount: number;\n  @Input() minActiveCount = 1;\n  @Input() resolveContext = true;\n  @Input() listTitle = '';\n  formArray: FormArray;\n  datapointLibraryEntries: Observable<IResultList<ManagedObjectKPI>>;\n  maxActiveCountReached = false;\n  AddButtonTypes = AddButtonTypes;\n\n  @Output() isValid: Observable<boolean>;\n  @Output() change: Observable<any[]>;\n  private usedValidators: { [key: string]: ValidatorFn } = {};\n\n  constructor(\n    private datapointSelector: DatapointSelectorService,\n    private datapointLibrary: DatapointLibraryService,\n    private formBuilder: FormBuilder,\n    @Optional() private widgetComponent: WidgetConfigComponent\n  ) {\n    this.formArray = this.formBuilder.array([]);\n    this.isValid = this.formArray.statusChanges.pipe(map(status => status === 'VALID'));\n    this.datapointLibraryEntries = from(this.datapointLibrary.getFirstDatapointLibraryPage()).pipe(\n      shareReplay()\n    );\n    this.change = this.formArray.valueChanges.pipe(map(res => this.transformValue(res)));\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (!changes.maxActiveCount && !changes.minActiveCount) {\n      return;\n    }\n    if (changes.maxActiveCount) {\n      this.usedValidators.maxActiveCount = C8yValidators.maxActiveCount(this.maxActiveCount);\n    }\n\n    if (changes.minActiveCount) {\n      this.usedValidators.minActiveCount = C8yValidators.minActiveCount(this.minActiveCount);\n    }\n    const validators = Object.values(this.usedValidators);\n    this.formArray.setValidators(validators);\n  }\n\n  registerOnTouched(fn: any): void {\n    this.formArray.valueChanges.pipe(take(1)).subscribe(fn);\n  }\n\n  validate(_control: AbstractControl): ValidationErrors {\n    return this.formArray.valid ? null : { formInvalid: {} };\n  }\n\n  ngOnInit(): void {\n    const context = this.widgetComponent?.context;\n    if (context?.id && this.resolveContext) {\n      const { name, id, c8y_IsDevice } = context;\n      this.config.contextAsset = { name, id, c8y_IsDevice };\n    }\n  }\n\n  writeValue(obj: KPIDetails[]): void {\n    this.formArray.clear();\n    if (obj?.length) {\n      obj.forEach(val => {\n        const formgroup = this.formBuilder.group({ details: [] });\n        formgroup.patchValue({ details: val });\n        this.formArray.push(formgroup);\n      });\n    }\n    this.calculateMaxActiveCount();\n  }\n\n  registerOnChange(fn: any): void {\n    this.formArray.valueChanges\n      .pipe(\n        map(res => this.transformValue(res)),\n        // check maxActiveCount\n        tap(() => {\n          this.calculateMaxActiveCount();\n        })\n      )\n      .subscribe(fn);\n  }\n\n  add() {\n    const allowChangingContext =\n      !this.widgetComponent?.isDeviceTypeDashboard && this.config?.allowChangingContext !== false;\n    this.datapointSelector\n      .selectDataPoints({\n        ...(this.config || {}),\n        selectedDatapoints: this.transformValue(this.formArray.value),\n        defaultActiveState: true,\n        allowChangingContext,\n        allowSearch: !this.config?.contextAsset\n      })\n      .then(\n        result => {\n          this.writeValue(result);\n        },\n        () => {\n          // nothing to do, modal was closed\n        }\n      );\n  }\n\n  onItemRemoved(index: number) {\n    this.formArray.removeAt(index);\n  }\n\n  drop(event: CdkDragDrop<KPIDetails[]>) {\n    const currentSorting = this.formArray.value;\n    moveItemInArray(currentSorting, event.previousIndex, event.currentIndex);\n    this.formArray.setValue(currentSorting);\n  }\n\n  private transformValue(formArrayValue: any[]) {\n    if (!formArrayValue) {\n      return [];\n    }\n    return formArrayValue.map(tmp => Object.assign({}, ...Object.values(tmp)));\n  }\n\n  private calculateMaxActiveCount() {\n    if (this.maxActiveCount) {\n      const currentlyActive = this.formArray.value.filter(tmp => tmp.details?.__active).length;\n      this.maxActiveCountReached = currentlyActive >= this.maxActiveCount;\n    }\n    this.maxActiveCountReached = false;\n  }\n}\n","<div class=\"card-header separator sticky-top bg-inherit\">\n  <span\n    class=\"card-title h4\"\n    *ngIf=\"listTitle\"\n  >\n    {{ listTitle | translate }}\n  </span>\n  <span\n    class=\"card-title h4\"\n    *ngIf=\"!listTitle\"\n  >\n    {{ 'Data points' | translate }}\n  </span>\n</div>\n\n<c8y-list-group\n  class=\"flex-grow ff-scroll-fix cdk-droplist\"\n  cdkDropList\n  (cdkDropListDropped)=\"drop($event)\"\n  [cdkDropListDisabled]=\"!allowDragAndDrop || formArray.controls?.length < 2\"\n>\n  <div\n    class=\"alert alert-warning m-t-8\"\n    role=\"alert\"\n    ngNonBindable\n    *ngIf=\"formArray.errors?.minActiveCount\"\n    translate\n    [translateParams]=\"formArray.errors?.minActiveCount\"\n  >\n    At least {{ minActive }} active data points must be selected.\n  </div>\n\n  <div\n    class=\"alert alert-warning m-t-8\"\n    role=\"alert\"\n    ngNonBindable\n    *ngIf=\"formArray.errors?.maxActiveCount\"\n    translate\n    [translateParams]=\"formArray.errors?.maxActiveCount\"\n  >\n    At maximum {{ maxActive }} active data points are allowed to be selected.\n  </div>\n\n  <ng-content select=\".alert\"></ng-content>\n\n  <div\n    class=\"p-t-8\"\n    *ngIf=\"!formArray.controls?.length\"\n  >\n    <c8y-ui-empty-state\n      class=\"p-t-8\"\n      [icon]=\"'c8y-data-points'\"\n      [title]=\"'No data points to display.' | translate\"\n      [subtitle]=\"'Add your first data point.' | translate\"\n      [horizontal]=\"true\"\n    ></c8y-ui-empty-state>\n  </div>\n  <div\n    [formGroup]=\"dpForm\"\n    *ngFor=\"let dpForm of formArray.controls; let index = index\"\n  >\n    <c8y-datapoint-selector-list-item\n      class=\"d-block\"\n      [defaultFormOptions]=\"defaultFormOptions\"\n      [activeToggleDisabled]=\"maxActiveCountReached\"\n      [showActiveToggle]=\"true\"\n      [addButtonType]=\"AddButtonTypes.none\"\n      [showOptions]=\"true\"\n      [editable]=\"true\"\n      [colorPickerDisabled]=\"false\"\n      [actions]=\"actions\"\n      [optionToRemove]=\"true\"\n      [datapointLibraryEntries]=\"datapointLibraryEntries\"\n      [hasUnlinkTemplateOption]=\"true\"\n      formControlName=\"details\"\n      (removed)=\"onItemRemoved(index)\"\n      cdkDrag\n    >\n      <c8y-li-drag-handle\n        title=\"{{ 'Click and drag to reorder' | translate }}\"\n        cdkDragHandle\n      >\n        <i c8yIcon=\"drag-reorder\"></i>\n      </c8y-li-drag-handle>\n    </c8y-datapoint-selector-list-item>\n  </div>\n</c8y-list-group>\n\n<div class=\"card-footer bg-inherit\">\n  <button\n    class=\"btn btn-default btn-sm\"\n    [title]=\"'Add data point' | translate\"\n    type=\"button\"\n    data-cy=\"c8y-datapoint-selection-list--add-datapoint-button\"\n    (click)=\"add()\"\n  >\n    <i c8yIcon=\"plus-circle\"></i>\n    {{ 'Add data point' | translate }}\n  </button>\n</div>\n"]}