@c8y/ngx-components 1021.57.0 → 1021.59.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/datapoint-explorer/view/configuration/name-generator.service.d.ts +7 -0
- package/datapoint-explorer/view/configuration/name-generator.service.d.ts.map +1 -0
- package/datapoint-explorer/view/configuration/workspace-configuration.component.d.ts +1 -0
- package/datapoint-explorer/view/configuration/workspace-configuration.component.d.ts.map +1 -1
- package/device-provisioned-certificates/device-tab-provisioned-certificates.component.d.ts.map +1 -1
- package/esm2022/datapoint-explorer/view/configuration/name-generator.service.mjs +20 -0
- package/esm2022/datapoint-explorer/view/configuration/workspace-configuration.component.mjs +4 -4
- package/esm2022/device-provisioned-certificates/device-provisioned-certificates.guard.mjs +2 -2
- package/esm2022/device-provisioned-certificates/device-tab-provisioned-certificates.component.mjs +7 -5
- package/esm2022/sub-assets/add-group/add-group.component.mjs +30 -4
- package/esm2022/sub-assets/add-group/add-group.module.mjs +23 -4
- package/esm2022/sub-assets/assign-devices/assign-child-devices.component.mjs +14 -2
- package/esm2022/sub-assets/groups.component.mjs +3 -3
- package/esm2022/sub-assets/sub-assets-grids.module.mjs +24 -0
- package/esm2022/sub-assets/sub-assets.component.mjs +4 -4
- package/esm2022/sub-assets/sub-assets.module.mjs +8 -12
- package/esm2022/trusted-certificates/list/trusted-certificate-list.component.mjs +5 -5
- package/fesm2022/c8y-ngx-components-datapoint-explorer-view.mjs +19 -2
- package/fesm2022/c8y-ngx-components-datapoint-explorer-view.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-device-provisioned-certificates.mjs +7 -5
- package/fesm2022/c8y-ngx-components-device-provisioned-certificates.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-sub-assets.mjs +1246 -1181
- package/fesm2022/c8y-ngx-components-sub-assets.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-trusted-certificates.mjs +4 -4
- package/fesm2022/c8y-ngx-components-trusted-certificates.mjs.map +1 -1
- package/locales/locales.pot +3 -3
- package/package.json +1 -1
- package/sub-assets/add-group/add-group.component.d.ts +9 -1
- package/sub-assets/add-group/add-group.component.d.ts.map +1 -1
- package/sub-assets/add-group/add-group.module.d.ts +3 -1
- package/sub-assets/add-group/add-group.module.d.ts.map +1 -1
- package/sub-assets/assign-devices/assign-child-devices.component.d.ts +3 -1
- package/sub-assets/assign-devices/assign-child-devices.component.d.ts.map +1 -1
- package/sub-assets/sub-assets-grids.module.d.ts +15 -0
- package/sub-assets/sub-assets-grids.module.d.ts.map +1 -0
- package/sub-assets/sub-assets.module.d.ts +18 -19
- package/sub-assets/sub-assets.module.d.ts.map +1 -1
- package/trusted-certificates/list/trusted-certificate-list.component.d.ts.map +1 -1
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import * as i0 from "@angular/core";
|
|
2
|
+
export declare class NameGeneratorService {
|
|
3
|
+
generateName(): string;
|
|
4
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<NameGeneratorService, never>;
|
|
5
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<NameGeneratorService>;
|
|
6
|
+
}
|
|
7
|
+
//# sourceMappingURL=name-generator.service.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"name-generator.service.d.ts","sourceRoot":"","sources":["../../../../datapoint-explorer/view/configuration/name-generator.service.ts"],"names":[],"mappings":";AAGA,qBAGa,oBAAoB;IAC/B,YAAY,IAAI,MAAM;yCADX,oBAAoB;6CAApB,oBAAoB;CAQhC"}
|
|
@@ -17,6 +17,7 @@ export declare class WorkspaceConfigComponent implements OnInit, OnChanges {
|
|
|
17
17
|
private readonly clipboardService;
|
|
18
18
|
private readonly activatedRoute;
|
|
19
19
|
private readonly router;
|
|
20
|
+
private readonly nameGeneratorService;
|
|
20
21
|
ngOnInit(): void;
|
|
21
22
|
ngOnChanges(changes: SimpleChanges): void;
|
|
22
23
|
addConfig(duplicatedConfig?: DatapointsGraphWidgetConfig): void;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"workspace-configuration.component.d.ts","sourceRoot":"","sources":["../../../../datapoint-explorer/view/configuration/workspace-configuration.component.ts"],"names":[],"mappings":"AACA,OAAO,EAEL,YAAY,EAGZ,SAAS,EACT,MAAM,EAEN,aAAa,EACd,MAAM,eAAe,CAAC;AACvB,OAAO,EAA0B,SAAS,EAAe,MAAM,gBAAgB,CAAC;AAUhF,OAAO,EAAE,sBAAsB,EAAE,MAAM,iCAAiC,CAAC;AACzE,OAAO,EAAE,2BAA2B,EAAE,MAAM,4BAA4B,CAAC;;
|
|
1
|
+
{"version":3,"file":"workspace-configuration.component.d.ts","sourceRoot":"","sources":["../../../../datapoint-explorer/view/configuration/workspace-configuration.component.ts"],"names":[],"mappings":"AACA,OAAO,EAEL,YAAY,EAGZ,SAAS,EACT,MAAM,EAEN,aAAa,EACd,MAAM,eAAe,CAAC;AACvB,OAAO,EAA0B,SAAS,EAAe,MAAM,gBAAgB,CAAC;AAUhF,OAAO,EAAE,sBAAsB,EAAE,MAAM,iCAAiC,CAAC;AACzE,OAAO,EAAE,2BAA2B,EAAE,MAAM,4BAA4B,CAAC;;AAKzE,qBAMa,wBAAyB,YAAW,MAAM,EAAE,SAAS;IACvD,aAAa,EAAE,2BAA2B,CAAC;IAC1C,qBAAqB,4CAAmD;IAElF,oBAAoB,EAAE,sBAAsB,CAAC;IAC7C,cAAc,EAAE,sBAAsB,EAAE,CAAM;IAC9C,uBAAuB,EAAE,SAAS,CAAC;IACnC,mBAAmB,2CAAqD;IACxE,mBAAmB,yBAAmC;IAEtD,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAuB;IACnD,OAAO,CAAC,QAAQ,CAAC,6BAA6B,CAAyC;IACvF,OAAO,CAAC,QAAQ,CAAC,mBAAmB,CAA+B;IACnE,OAAO,CAAC,QAAQ,CAAC,gBAAgB,CAA4B;IAC7D,OAAO,CAAC,QAAQ,CAAC,cAAc,CAA0B;IACzD,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAkB;IACzC,OAAO,CAAC,QAAQ,CAAC,oBAAoB,CAAgC;IAErE,QAAQ,IAAI,IAAI;IAMhB,WAAW,CAAC,OAAO,EAAE,aAAa,GAAG,IAAI;IAMzC,SAAS,CAAC,gBAAgB,CAAC,EAAE,2BAA2B,GAAG,IAAI;IAe/D,mBAAmB,CAAC,QAAQ,EAAE,OAAO,EAAE,aAAa,EAAE,sBAAsB,GAAG,IAAI;IAYnF,wBAAwB,CAAC,aAAa,EAAE,sBAAsB,GAAG,IAAI;IAYrE,mBAAmB,CAAC,aAAa,EAAE,sBAAsB,GAAG,IAAI;IAShE,QAAQ,IAAI,IAAI;IASV,WAAW,CAAC,aAAa,EAAE,sBAAsB,GAAG,OAAO,CAAC,IAAI,CAAC;IAIvE,OAAO,CAAC,gBAAgB;IAMxB,OAAO,CAAC,oBAAoB;IA0B5B,OAAO,CAAC,wBAAwB;IA+BhC,OAAO,CAAC,iBAAiB;IAQzB,OAAO,CAAC,SAAS;IAOjB,OAAO,CAAC,WAAW;IAQnB,OAAO,CAAC,yBAAyB;yCAjLtB,wBAAwB;2CAAxB,wBAAwB;CA2LpC"}
|
package/device-provisioned-certificates/device-tab-provisioned-certificates.component.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"device-tab-provisioned-certificates.component.d.ts","sourceRoot":"","sources":["../../device-provisioned-certificates/device-tab-provisioned-certificates.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAa,MAAM,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,YAAY,EAAW,YAAY,EAAsB,MAAM,qBAAqB,CAAC;AAC9F,OAAO,EAAE,oCAAoC,EAAE,MAAM,2CAA2C,CAAC;AACjG,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,UAAU,EAAkB,MAAM,aAAa,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AAEnD,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;;AAEvD,MAAM,MAAM,sBAAsB,GAAG;IAAE,YAAY,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,CAAC;AAEjF,qBAIa,yCAA0C,YAAW,MAAM;
|
|
1
|
+
{"version":3,"file":"device-tab-provisioned-certificates.component.d.ts","sourceRoot":"","sources":["../../device-provisioned-certificates/device-tab-provisioned-certificates.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAa,MAAM,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,YAAY,EAAW,YAAY,EAAsB,MAAM,qBAAqB,CAAC;AAC9F,OAAO,EAAE,oCAAoC,EAAE,MAAM,2CAA2C,CAAC;AACjG,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,UAAU,EAAkB,MAAM,aAAa,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AAEnD,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;;AAEvD,MAAM,MAAM,sBAAsB,GAAG;IAAE,YAAY,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,CAAC;AAEjF,qBAIa,yCAA0C,YAAW,MAAM;IAqBpE,OAAO,CAAC,YAAY;IACpB,OAAO,CAAC,YAAY;IACpB,OAAO,CAAC,oBAAoB;IAC5B,OAAO,CAAC,UAAU;IAClB,OAAO,CAAC,KAAK;IACb,OAAO,CAAC,gBAAgB;IAzB1B,SAAS,UAAS;IAClB,MAAM,EAAE,eAAe,CAAC,IAAI,CAAC,CAA6B;IAC1D,uBAAuB,EAAE,UAAU,CAAC,sBAAsB,EAAE,CAAC,CAa3D;IAEF,SAAS,CAAC,QAAQ,CAAC,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;MAAc;gBAGjC,YAAY,EAAE,YAAY,EAC1B,YAAY,EAAE,YAAY,EAC1B,oBAAoB,EAAE,oCAAoC,EAC1D,UAAU,EAAE,UAAU,EACtB,KAAK,EAAE,cAAc,EACrB,gBAAgB,EAAE,gBAAgB;IAG5C,QAAQ,IAAI,IAAI;IAIhB,oCAAoC,CAAC,IAAI,EAAE,sBAAsB;IAgB3D,4BAA4B,CAAC,sBAAsB,EAAE,sBAAsB;YAcnE,iBAAiB;yCA/DpB,yCAAyC;2CAAzC,yCAAyC;CA4ErD"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { adjectives, nouns } from './naming-dictionary';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class NameGeneratorService {
|
|
5
|
+
generateName() {
|
|
6
|
+
const getRandomElement = (arr) => arr[Math.floor(Math.random() * arr.length)];
|
|
7
|
+
const randomAdjective = getRandomElement(adjectives);
|
|
8
|
+
const randomNoun = getRandomElement(nouns);
|
|
9
|
+
return `${randomAdjective}_${randomNoun}`;
|
|
10
|
+
}
|
|
11
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NameGeneratorService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
12
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NameGeneratorService, providedIn: 'root' }); }
|
|
13
|
+
}
|
|
14
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NameGeneratorService, decorators: [{
|
|
15
|
+
type: Injectable,
|
|
16
|
+
args: [{
|
|
17
|
+
providedIn: 'root'
|
|
18
|
+
}]
|
|
19
|
+
}] });
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmFtZS1nZW5lcmF0b3Iuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2RhdGFwb2ludC1leHBsb3Jlci92aWV3L2NvbmZpZ3VyYXRpb24vbmFtZS1nZW5lcmF0b3Iuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNDLE9BQU8sRUFBRSxVQUFVLEVBQUUsS0FBSyxFQUFFLE1BQU0scUJBQXFCLENBQUM7O0FBS3hELE1BQU0sT0FBTyxvQkFBb0I7SUFDL0IsWUFBWTtRQUNWLE1BQU0sZ0JBQWdCLEdBQUcsQ0FBQyxHQUFhLEVBQUUsRUFBRSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxHQUFHLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQztRQUN4RixNQUFNLGVBQWUsR0FBRyxnQkFBZ0IsQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUNyRCxNQUFNLFVBQVUsR0FBRyxnQkFBZ0IsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUUzQyxPQUFPLEdBQUcsZUFBZSxJQUFJLFVBQVUsRUFBRSxDQUFDO0lBQzVDLENBQUM7K0dBUFUsb0JBQW9CO21IQUFwQixvQkFBb0IsY0FGbkIsTUFBTTs7NEZBRVAsb0JBQW9CO2tCQUhoQyxVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IGFkamVjdGl2ZXMsIG5vdW5zIH0gZnJvbSAnLi9uYW1pbmctZGljdGlvbmFyeSc7XG5cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXG59KVxuZXhwb3J0IGNsYXNzIE5hbWVHZW5lcmF0b3JTZXJ2aWNlIHtcbiAgZ2VuZXJhdGVOYW1lKCk6IHN0cmluZyB7XG4gICAgY29uc3QgZ2V0UmFuZG9tRWxlbWVudCA9IChhcnI6IHN0cmluZ1tdKSA9PiBhcnJbTWF0aC5mbG9vcihNYXRoLnJhbmRvbSgpICogYXJyLmxlbmd0aCldO1xuICAgIGNvbnN0IHJhbmRvbUFkamVjdGl2ZSA9IGdldFJhbmRvbUVsZW1lbnQoYWRqZWN0aXZlcyk7XG4gICAgY29uc3QgcmFuZG9tTm91biA9IGdldFJhbmRvbUVsZW1lbnQobm91bnMpO1xuXG4gICAgcmV0dXJuIGAke3JhbmRvbUFkamVjdGl2ZX1fJHtyYW5kb21Ob3VufWA7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -6,8 +6,7 @@ import { BsDropdownModule } from 'ngx-bootstrap/dropdown';
|
|
|
6
6
|
import { TooltipModule } from 'ngx-bootstrap/tooltip';
|
|
7
7
|
import { WorkspaceConfigurationService } from './workspace-configuration.service';
|
|
8
8
|
import { ActivatedRoute, Router } from '@angular/router';
|
|
9
|
-
import {
|
|
10
|
-
import { adjectives, nouns } from './naming-dictionary';
|
|
9
|
+
import { NameGeneratorService } from './name-generator.service';
|
|
11
10
|
import * as i0 from "@angular/core";
|
|
12
11
|
import * as i1 from "@c8y/ngx-components";
|
|
13
12
|
import * as i2 from "@angular/common";
|
|
@@ -27,6 +26,7 @@ export class WorkspaceConfigComponent {
|
|
|
27
26
|
this.clipboardService = inject(ClipboardService);
|
|
28
27
|
this.activatedRoute = inject(ActivatedRoute);
|
|
29
28
|
this.router = inject(Router);
|
|
29
|
+
this.nameGeneratorService = inject(NameGeneratorService);
|
|
30
30
|
}
|
|
31
31
|
ngOnInit() {
|
|
32
32
|
this.initializeContextSourceId();
|
|
@@ -39,7 +39,7 @@ export class WorkspaceConfigComponent {
|
|
|
39
39
|
}
|
|
40
40
|
}
|
|
41
41
|
addConfig(duplicatedConfig) {
|
|
42
|
-
const name =
|
|
42
|
+
const name = this.nameGeneratorService.generateName();
|
|
43
43
|
const workspace = {
|
|
44
44
|
id: new Date().toISOString(),
|
|
45
45
|
label: name,
|
|
@@ -171,4 +171,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
171
171
|
}], onConfigurationChange: [{
|
|
172
172
|
type: Output
|
|
173
173
|
}] } });
|
|
174
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workspace-configuration.component.js","sourceRoot":"","sources":["../../../../../datapoint-explorer/view/configuration/workspace-configuration.component.ts","../../../../../datapoint-explorer/view/configuration/workspace-configuration.component.html"],"names":[],"mappings":"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","sourcesContent":["import { A11yModule } from '@angular/cdk/a11y';\nimport {\n  Component,\n  EventEmitter,\n  inject,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges\n} from '@angular/core';\nimport { FormArray, FormBuilder, FormGroup, FormsModule } from '@angular/forms';\nimport {\n  ClipboardService,\n  ContextRouteService,\n  CoreModule,\n  gettext,\n  ViewContext\n} from '@c8y/ngx-components';\nimport { BsDropdownModule } from 'ngx-bootstrap/dropdown';\nimport { TooltipModule } from 'ngx-bootstrap/tooltip';\nimport { WorkspaceConfiguration } from './workspace-configuration.model';\nimport { DatapointsGraphWidgetConfig } from '@c8y/ngx-components/echart';\nimport { WorkspaceConfigurationService } from './workspace-configuration.service';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { uniqueNamesGenerator } from 'unique-names-generator';\nimport { adjectives, nouns } from './naming-dictionary';\n\n@Component({\n  selector: 'c8y-workspace-config',\n  templateUrl: './workspace-configuration.component.html',\n  standalone: true,\n  imports: [CoreModule, BsDropdownModule, FormsModule, A11yModule, TooltipModule]\n})\nexport class WorkspaceConfigComponent implements OnInit, OnChanges {\n  @Input() updatedConfig: DatapointsGraphWidgetConfig;\n  @Output() onConfigurationChange = new EventEmitter<DatapointsGraphWidgetConfig>();\n\n  currentConfiguration: WorkspaceConfiguration;\n  configurations: WorkspaceConfiguration[] = [];\n  configurationsFormGroup: FormGroup;\n  activeConfigTooltip = gettext('Active configuration cannot be removed');\n  removeConfigTooltip = gettext('Remove configuration');\n\n  private readonly formBuilder = inject(FormBuilder);\n  private readonly workspaceConfigurationService = inject(WorkspaceConfigurationService);\n  private readonly contextRouteService = inject(ContextRouteService);\n  private readonly clipboardService = inject(ClipboardService);\n  private readonly activatedRoute = inject(ActivatedRoute);\n  private readonly router = inject(Router);\n\n  ngOnInit(): void {\n    this.initializeContextSourceId();\n    this.initializeConfigurations();\n    this.initWorkspaceForm();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.updatedConfig && !changes.updatedConfig.firstChange) {\n      this.updateConfigurations();\n    }\n  }\n\n  addConfig(duplicatedConfig?: DatapointsGraphWidgetConfig): void {\n    const name = uniqueNamesGenerator({ dictionaries: [adjectives, nouns] });\n    const workspace: WorkspaceConfiguration = {\n      id: new Date().toISOString(),\n      label: name,\n      config: { datapoints: [], alarmsEventsConfigs: [] }\n    };\n    if (duplicatedConfig) {\n      workspace.config = duplicatedConfig;\n    }\n    this.configurations = [workspace, ...this.configurations];\n    this.initWorkspaceForm();\n    this.changeConfiguration(true, workspace);\n  }\n\n  changeConfiguration(selected: boolean, configuration: WorkspaceConfiguration): void {\n    if (!selected) {\n      return;\n    }\n    this.currentConfiguration = configuration;\n    localStorage.setItem(\n      this.workspaceConfigurationService.LOCAL_STORAGE_DEFAULT_ID_KEY,\n      this.currentConfiguration.id\n    );\n    this.onConfigurationChange.emit(configuration.config);\n  }\n\n  updateConfigurationLabel(configuration: WorkspaceConfiguration): void {\n    this.configurations = this.configurations.map(c =>\n      c.id === configuration.id ? configuration : c\n    );\n    this.workspaceConfigurationService.saveConfigurations(\n      this.configurations,\n      this.currentConfiguration?.id || ''\n    );\n    this.currentConfiguration = configuration;\n    this.onConfigurationChange.emit(configuration.config);\n  }\n\n  deleteConfiguration(configuration: WorkspaceConfiguration): void {\n    this.configurations = this.configurations.filter(c => c.id !== configuration.id);\n    this.initWorkspaceForm();\n    this.workspaceConfigurationService.saveConfigurations(\n      this.configurations,\n      this.currentConfiguration?.id || ''\n    );\n  }\n\n  clearAll(): void {\n    this.configurations = [this.currentConfiguration];\n    this.initWorkspaceForm();\n    this.workspaceConfigurationService.saveConfigurations(\n      this.configurations,\n      this.currentConfiguration?.id || ''\n    );\n  }\n\n  async shareConfig(configuration: WorkspaceConfiguration): Promise<void> {\n    await this.clipboardService.writeText(JSON.stringify(configuration.config));\n  }\n\n  private addConfigFromUrl(queryParams: any): void {\n    this.addConfig();\n    const config = JSON.parse(queryParams.config);\n    this.onConfigurationChange.emit(config);\n  }\n\n  private updateConfigurations(): void {\n    this.currentConfiguration.config = this.updatedConfig;\n    this.configurations = this.configurations.map(currentConfig =>\n      currentConfig.id === this.currentConfiguration.id ? this.currentConfiguration : currentConfig\n    );\n    const queryParams = {\n      configId: this.currentConfiguration.id,\n      label: this.currentConfiguration.label,\n      config: JSON.stringify(this.updatedConfig)\n    };\n    this.workspaceConfigurationService.saveConfigurations(\n      this.configurations,\n      this.currentConfiguration.id\n    );\n    const control = <FormArray>this.configurationsFormGroup.controls['configurations'];\n    const index = this.configurations.findIndex(c => c.id === this.currentConfiguration.id);\n    if (index !== -1) {\n      control.at(index).patchValue({\n        label: this.currentConfiguration.label,\n        config: this.currentConfiguration.config,\n        id: this.currentConfiguration.id\n      });\n    }\n    this.router.navigate([], { queryParams });\n  }\n\n  private initializeConfigurations(): void {\n    const configurations = this.workspaceConfigurationService.getConfigurations();\n    const defaultId = this.workspaceConfigurationService.getDefaultConfigurationId();\n    const queryParams = this.router.parseUrl(this.router.url).queryParams;\n    if (configurations.length) {\n      this.configurations = configurations;\n      this.currentConfiguration =\n        this.configurations.find(c => c.id === defaultId) || this.configurations[0];\n\n      this.onConfigurationChange.emit(this.currentConfiguration.config);\n      if (queryParams?.configId && !this.configurations.find(c => c.id === queryParams.configId)) {\n        this.addConfigFromUrl(queryParams);\n      } else if (\n        queryParams?.configId &&\n        this.configurations.find(c => c.id === queryParams.configId) &&\n        this.currentConfiguration.id !== queryParams.configId\n      ) {\n        this.currentConfiguration = this.configurations.find(c => c.id === queryParams.configId);\n        this.changeConfiguration(true, this.currentConfiguration);\n      }\n    }\n\n    if (!this.currentConfiguration) {\n      if (Object.keys(queryParams).length === 0) {\n        this.addConfig();\n        return;\n      }\n      this.addConfigFromUrl(queryParams);\n    }\n  }\n\n  private initWorkspaceForm(): void {\n    this.configurationsFormGroup = this.formBuilder.group({\n      configurations: this.formBuilder.array([])\n    });\n\n    this.patchForm();\n  }\n\n  private patchForm(): void {\n    const control = <FormArray>this.configurationsFormGroup.controls['configurations'];\n    this.configurations.forEach(workspace => {\n      control.push(this.patchValues(workspace));\n    });\n  }\n\n  private patchValues(workspace: WorkspaceConfiguration): FormGroup {\n    return this.formBuilder.group({\n      label: [workspace.label],\n      config: [workspace.config],\n      id: [workspace.id]\n    });\n  }\n\n  private initializeContextSourceId(): void {\n    const routeContext = this.contextRouteService.getContextData(this.activatedRoute);\n    if (!routeContext) {\n      return;\n    }\n    const { context, contextData } = routeContext;\n    if ([ViewContext.Device, ViewContext.Group].includes(context)) {\n      this.workspaceConfigurationService.contextIdSignal.set(contextData?.id);\n    }\n  }\n}\n","<div\n  class=\"dropdown\"\n  #actionbar_dropdown=\"bs-dropdown\"\n  [cdkTrapFocus]=\"actionbar_dropdown.isOpen\"\n  dropdown\n  [insideClick]=\"true\"\n>\n\n  <button\n    class=\"dropdown-toggle form-control l-h-tight d-flex a-i-center\"\n    attr.aria-label=\"{{ currentConfiguration.label }}\"\n    tooltip=\"{{ 'Selected configuration' | translate }}\"\n    placement=\"top\"\n    container=\"body\"\n    data-cy=\"current-configuration-dropdown-button\"\n    [adaptivePosition]=\"false\"\n    [delay]=\"500\"\n    dropdownToggle\n  >\n    <i\n      class=\"m-r-4\"\n      c8yIcon=\"list\"\n    ></i>\n    <div class=\"d-col text-left fit-w\">\n      <span class=\"text-12\">\n        {{ 'Configuration' | translate }}\n      </span>\n      <span class=\"text-10 text-muted text-truncate\">\n        {{ currentConfiguration.label }}\n      </span>\n    </div>\n    <span class=\"caret m-r-16 m-l-4\"></span>\n  </button>\n  <div\n    class=\"dropdown-menu dropdown-menu-wide dropdown-menu-action-bar\"\n    *dropdownMenu\n  >\n    <div class=\"sticky-top separator-bottom p-t-8 p-b-8 p-l-16 p-r-16\">\n      <p>\n        <strong>{{ 'Data explorer configurations' | translate }}</strong>\n      </p>\n      <p>\n        <small>{{ 'Easily switch and manage configurations.' | translate }}</small>\n      </p>\n    </div>\n    <c8y-list-group class=\"no-border-last\">\n      <form [formGroup]=\"configurationsFormGroup\">\n        <div formArrayName=\"configurations\">\n          <c8y-li\n            class=\"p-0\"\n            *ngFor=\"\n              let configuration of configurationsFormGroup.controls.configurations['controls'];\n              let i = index\n            \"\n            [dense]=\"true\"\n          >\n            <c8y-li-radio\n              [selected]=\"configuration.value.id === currentConfiguration.id\"\n              (onSelect)=\"changeConfiguration($event, configuration.value)\"\n            ></c8y-li-radio>\n            <div class=\"d-flex a-i-center gap-8\">\n              <div\n                class=\"flex-grow min-width-0\"\n                [formGroupName]=\"i\"\n              >\n                <label\n                  class=\"editable\"\n                  [ngClass]=\"{\n                    updated:\n                      configuration.controls.label.touched && configuration.controls.label.dirty\n                  }\"\n                >\n                  <input\n                    class=\"form-control\"\n                    [style.width.ch]=\"configuration.value.label || 25\"\n                    [attr.aria-label]=\"'Configuration label' | translate\"\n                    placeholder=\"{{ 'Configuration 1' | translate }}\"\n                    type=\"text\"\n                    autocomplete=\"off\"\n                    required\n                    formControlName=\"label\"\n                    (blur)=\"updateConfigurationLabel(configuration.value)\"\n                  />\n                </label>\n              </div>\n              <div class=\"flex-nogrow d-flex gap-8\">\n                <button\n                  class=\"btn-dot btn\"\n                  [attr.aria-label]=\"'Duplicate configuration' | translate\"\n                  tooltip=\"{{ 'Duplicate configuration' | translate }}\"\n                  placement=\"left\"\n                  (click)=\"addConfig(configuration.value.config)\"\n                  [delay]=\"500\"\n                >\n                  <i c8yIcon=\"copy\"></i>\n                </button>\n\n                <button\n                  class=\"btn-dot btn btn-dot--danger\"\n                  [attr.aria-label]=\"'Remove configurations' | translate\"\n                  tooltip=\"{{\n                    (configuration.value.id === currentConfiguration.id\n                      ? activeConfigTooltip\n                      : removeConfigTooltip\n                    ) | translate\n                  }}\"\n                  placement=\"left\"\n                  [delay]=\"500\"\n                  (click)=\"$event.stopPropagation(); deleteConfiguration(configuration.value)\"\n                  [disabled]=\"configuration.value.id === currentConfiguration.id\"\n                >\n                  <i c8yIcon=\"minus-circle\"></i>\n                </button>\n              </div>\n            </div>\n          </c8y-li>\n        </div>\n      </form>\n    </c8y-list-group>\n    <div class=\"sticky-bottom separator-top\">\n      <div class=\"d-flex p-l-16 p-r-16 p-t-8 p-b-8\">\n        <button\n          class=\"btn btn-danger btn-sm flex-grow\"\n          (click)=\"clearAll()\"\n          [disabled]=\"configurations.length < 2\"\n        >\n          <i [c8yIcon]=\"'delete'\"></i>\n          {{ 'Delete all configurations' | translate }}\n        </button>\n        <button\n          class=\"btn btn-default btn-sm flex-grow\"\n          type=\"button\"\n          (click)=\"addConfig()\"\n        >\n          <i [c8yIcon]=\"'add-circle-outline'\"></i>\n          {{ 'Add configuration' | translate }}\n        </button>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
174
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workspace-configuration.component.js","sourceRoot":"","sources":["../../../../../datapoint-explorer/view/configuration/workspace-configuration.component.ts","../../../../../datapoint-explorer/view/configuration/workspace-configuration.component.html"],"names":[],"mappings":"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","sourcesContent":["import { A11yModule } from '@angular/cdk/a11y';\nimport {\n  Component,\n  EventEmitter,\n  inject,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges\n} from '@angular/core';\nimport { FormArray, FormBuilder, FormGroup, FormsModule } from '@angular/forms';\nimport {\n  ClipboardService,\n  ContextRouteService,\n  CoreModule,\n  gettext,\n  ViewContext\n} from '@c8y/ngx-components';\nimport { BsDropdownModule } from 'ngx-bootstrap/dropdown';\nimport { TooltipModule } from 'ngx-bootstrap/tooltip';\nimport { WorkspaceConfiguration } from './workspace-configuration.model';\nimport { DatapointsGraphWidgetConfig } from '@c8y/ngx-components/echart';\nimport { WorkspaceConfigurationService } from './workspace-configuration.service';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { NameGeneratorService } from './name-generator.service';\n\n@Component({\n  selector: 'c8y-workspace-config',\n  templateUrl: './workspace-configuration.component.html',\n  standalone: true,\n  imports: [CoreModule, BsDropdownModule, FormsModule, A11yModule, TooltipModule]\n})\nexport class WorkspaceConfigComponent implements OnInit, OnChanges {\n  @Input() updatedConfig: DatapointsGraphWidgetConfig;\n  @Output() onConfigurationChange = new EventEmitter<DatapointsGraphWidgetConfig>();\n\n  currentConfiguration: WorkspaceConfiguration;\n  configurations: WorkspaceConfiguration[] = [];\n  configurationsFormGroup: FormGroup;\n  activeConfigTooltip = gettext('Active configuration cannot be removed');\n  removeConfigTooltip = gettext('Remove configuration');\n\n  private readonly formBuilder = inject(FormBuilder);\n  private readonly workspaceConfigurationService = inject(WorkspaceConfigurationService);\n  private readonly contextRouteService = inject(ContextRouteService);\n  private readonly clipboardService = inject(ClipboardService);\n  private readonly activatedRoute = inject(ActivatedRoute);\n  private readonly router = inject(Router);\n  private readonly nameGeneratorService = inject(NameGeneratorService);\n\n  ngOnInit(): void {\n    this.initializeContextSourceId();\n    this.initializeConfigurations();\n    this.initWorkspaceForm();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.updatedConfig && !changes.updatedConfig.firstChange) {\n      this.updateConfigurations();\n    }\n  }\n\n  addConfig(duplicatedConfig?: DatapointsGraphWidgetConfig): void {\n    const name = this.nameGeneratorService.generateName();\n    const workspace: WorkspaceConfiguration = {\n      id: new Date().toISOString(),\n      label: name,\n      config: { datapoints: [], alarmsEventsConfigs: [] }\n    };\n    if (duplicatedConfig) {\n      workspace.config = duplicatedConfig;\n    }\n    this.configurations = [workspace, ...this.configurations];\n    this.initWorkspaceForm();\n    this.changeConfiguration(true, workspace);\n  }\n\n  changeConfiguration(selected: boolean, configuration: WorkspaceConfiguration): void {\n    if (!selected) {\n      return;\n    }\n    this.currentConfiguration = configuration;\n    localStorage.setItem(\n      this.workspaceConfigurationService.LOCAL_STORAGE_DEFAULT_ID_KEY,\n      this.currentConfiguration.id\n    );\n    this.onConfigurationChange.emit(configuration.config);\n  }\n\n  updateConfigurationLabel(configuration: WorkspaceConfiguration): void {\n    this.configurations = this.configurations.map(c =>\n      c.id === configuration.id ? configuration : c\n    );\n    this.workspaceConfigurationService.saveConfigurations(\n      this.configurations,\n      this.currentConfiguration?.id || ''\n    );\n    this.currentConfiguration = configuration;\n    this.onConfigurationChange.emit(configuration.config);\n  }\n\n  deleteConfiguration(configuration: WorkspaceConfiguration): void {\n    this.configurations = this.configurations.filter(c => c.id !== configuration.id);\n    this.initWorkspaceForm();\n    this.workspaceConfigurationService.saveConfigurations(\n      this.configurations,\n      this.currentConfiguration?.id || ''\n    );\n  }\n\n  clearAll(): void {\n    this.configurations = [this.currentConfiguration];\n    this.initWorkspaceForm();\n    this.workspaceConfigurationService.saveConfigurations(\n      this.configurations,\n      this.currentConfiguration?.id || ''\n    );\n  }\n\n  async shareConfig(configuration: WorkspaceConfiguration): Promise<void> {\n    await this.clipboardService.writeText(JSON.stringify(configuration.config));\n  }\n\n  private addConfigFromUrl(queryParams: any): void {\n    this.addConfig();\n    const config = JSON.parse(queryParams.config);\n    this.onConfigurationChange.emit(config);\n  }\n\n  private updateConfigurations(): void {\n    this.currentConfiguration.config = this.updatedConfig;\n    this.configurations = this.configurations.map(currentConfig =>\n      currentConfig.id === this.currentConfiguration.id ? this.currentConfiguration : currentConfig\n    );\n    const queryParams = {\n      configId: this.currentConfiguration.id,\n      label: this.currentConfiguration.label,\n      config: JSON.stringify(this.updatedConfig)\n    };\n    this.workspaceConfigurationService.saveConfigurations(\n      this.configurations,\n      this.currentConfiguration.id\n    );\n    const control = <FormArray>this.configurationsFormGroup.controls['configurations'];\n    const index = this.configurations.findIndex(c => c.id === this.currentConfiguration.id);\n    if (index !== -1) {\n      control.at(index).patchValue({\n        label: this.currentConfiguration.label,\n        config: this.currentConfiguration.config,\n        id: this.currentConfiguration.id\n      });\n    }\n    this.router.navigate([], { queryParams });\n  }\n\n  private initializeConfigurations(): void {\n    const configurations = this.workspaceConfigurationService.getConfigurations();\n    const defaultId = this.workspaceConfigurationService.getDefaultConfigurationId();\n    const queryParams = this.router.parseUrl(this.router.url).queryParams;\n    if (configurations.length) {\n      this.configurations = configurations;\n      this.currentConfiguration =\n        this.configurations.find(c => c.id === defaultId) || this.configurations[0];\n\n      this.onConfigurationChange.emit(this.currentConfiguration.config);\n      if (queryParams?.configId && !this.configurations.find(c => c.id === queryParams.configId)) {\n        this.addConfigFromUrl(queryParams);\n      } else if (\n        queryParams?.configId &&\n        this.configurations.find(c => c.id === queryParams.configId) &&\n        this.currentConfiguration.id !== queryParams.configId\n      ) {\n        this.currentConfiguration = this.configurations.find(c => c.id === queryParams.configId);\n        this.changeConfiguration(true, this.currentConfiguration);\n      }\n    }\n\n    if (!this.currentConfiguration) {\n      if (Object.keys(queryParams).length === 0) {\n        this.addConfig();\n        return;\n      }\n      this.addConfigFromUrl(queryParams);\n    }\n  }\n\n  private initWorkspaceForm(): void {\n    this.configurationsFormGroup = this.formBuilder.group({\n      configurations: this.formBuilder.array([])\n    });\n\n    this.patchForm();\n  }\n\n  private patchForm(): void {\n    const control = <FormArray>this.configurationsFormGroup.controls['configurations'];\n    this.configurations.forEach(workspace => {\n      control.push(this.patchValues(workspace));\n    });\n  }\n\n  private patchValues(workspace: WorkspaceConfiguration): FormGroup {\n    return this.formBuilder.group({\n      label: [workspace.label],\n      config: [workspace.config],\n      id: [workspace.id]\n    });\n  }\n\n  private initializeContextSourceId(): void {\n    const routeContext = this.contextRouteService.getContextData(this.activatedRoute);\n    if (!routeContext) {\n      return;\n    }\n    const { context, contextData } = routeContext;\n    if ([ViewContext.Device, ViewContext.Group].includes(context)) {\n      this.workspaceConfigurationService.contextIdSignal.set(contextData?.id);\n    }\n  }\n}\n","<div\n  class=\"dropdown\"\n  #actionbar_dropdown=\"bs-dropdown\"\n  [cdkTrapFocus]=\"actionbar_dropdown.isOpen\"\n  dropdown\n  [insideClick]=\"true\"\n>\n\n  <button\n    class=\"dropdown-toggle form-control l-h-tight d-flex a-i-center\"\n    attr.aria-label=\"{{ currentConfiguration.label }}\"\n    tooltip=\"{{ 'Selected configuration' | translate }}\"\n    placement=\"top\"\n    container=\"body\"\n    data-cy=\"current-configuration-dropdown-button\"\n    [adaptivePosition]=\"false\"\n    [delay]=\"500\"\n    dropdownToggle\n  >\n    <i\n      class=\"m-r-4\"\n      c8yIcon=\"list\"\n    ></i>\n    <div class=\"d-col text-left fit-w\">\n      <span class=\"text-12\">\n        {{ 'Configuration' | translate }}\n      </span>\n      <span class=\"text-10 text-muted text-truncate\">\n        {{ currentConfiguration.label }}\n      </span>\n    </div>\n    <span class=\"caret m-r-16 m-l-4\"></span>\n  </button>\n  <div\n    class=\"dropdown-menu dropdown-menu-wide dropdown-menu-action-bar\"\n    *dropdownMenu\n  >\n    <div class=\"sticky-top separator-bottom p-t-8 p-b-8 p-l-16 p-r-16\">\n      <p>\n        <strong>{{ 'Data explorer configurations' | translate }}</strong>\n      </p>\n      <p>\n        <small>{{ 'Easily switch and manage configurations.' | translate }}</small>\n      </p>\n    </div>\n    <c8y-list-group class=\"no-border-last\">\n      <form [formGroup]=\"configurationsFormGroup\">\n        <div formArrayName=\"configurations\">\n          <c8y-li\n            class=\"p-0\"\n            *ngFor=\"\n              let configuration of configurationsFormGroup.controls.configurations['controls'];\n              let i = index\n            \"\n            [dense]=\"true\"\n          >\n            <c8y-li-radio\n              [selected]=\"configuration.value.id === currentConfiguration.id\"\n              (onSelect)=\"changeConfiguration($event, configuration.value)\"\n            ></c8y-li-radio>\n            <div class=\"d-flex a-i-center gap-8\">\n              <div\n                class=\"flex-grow min-width-0\"\n                [formGroupName]=\"i\"\n              >\n                <label\n                  class=\"editable\"\n                  [ngClass]=\"{\n                    updated:\n                      configuration.controls.label.touched && configuration.controls.label.dirty\n                  }\"\n                >\n                  <input\n                    class=\"form-control\"\n                    [style.width.ch]=\"configuration.value.label || 25\"\n                    [attr.aria-label]=\"'Configuration label' | translate\"\n                    placeholder=\"{{ 'Configuration 1' | translate }}\"\n                    type=\"text\"\n                    autocomplete=\"off\"\n                    required\n                    formControlName=\"label\"\n                    (blur)=\"updateConfigurationLabel(configuration.value)\"\n                  />\n                </label>\n              </div>\n              <div class=\"flex-nogrow d-flex gap-8\">\n                <button\n                  class=\"btn-dot btn\"\n                  [attr.aria-label]=\"'Duplicate configuration' | translate\"\n                  tooltip=\"{{ 'Duplicate configuration' | translate }}\"\n                  placement=\"left\"\n                  (click)=\"addConfig(configuration.value.config)\"\n                  [delay]=\"500\"\n                >\n                  <i c8yIcon=\"copy\"></i>\n                </button>\n\n                <button\n                  class=\"btn-dot btn btn-dot--danger\"\n                  [attr.aria-label]=\"'Remove configurations' | translate\"\n                  tooltip=\"{{\n                    (configuration.value.id === currentConfiguration.id\n                      ? activeConfigTooltip\n                      : removeConfigTooltip\n                    ) | translate\n                  }}\"\n                  placement=\"left\"\n                  [delay]=\"500\"\n                  (click)=\"$event.stopPropagation(); deleteConfiguration(configuration.value)\"\n                  [disabled]=\"configuration.value.id === currentConfiguration.id\"\n                >\n                  <i c8yIcon=\"minus-circle\"></i>\n                </button>\n              </div>\n            </div>\n          </c8y-li>\n        </div>\n      </form>\n    </c8y-list-group>\n    <div class=\"sticky-bottom separator-top\">\n      <div class=\"d-flex p-l-16 p-r-16 p-t-8 p-b-8\">\n        <button\n          class=\"btn btn-danger btn-sm flex-grow\"\n          (click)=\"clearAll()\"\n          [disabled]=\"configurations.length < 2\"\n        >\n          <i [c8yIcon]=\"'delete'\"></i>\n          {{ 'Delete all configurations' | translate }}\n        </button>\n        <button\n          class=\"btn btn-default btn-sm flex-grow\"\n          type=\"button\"\n          (click)=\"addConfig()\"\n        >\n          <i [c8yIcon]=\"'add-circle-outline'\"></i>\n          {{ 'Add configuration' | translate }}\n        </button>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -13,7 +13,7 @@ export class DeviceProvisionedCertificatesGuard {
|
|
|
13
13
|
}
|
|
14
14
|
try {
|
|
15
15
|
const provisionedCertificates = await this.deviceProvisionedCertificatesService.getDeviceProvisionedCertificates(contextData);
|
|
16
|
-
return
|
|
16
|
+
return provisionedCertificates && provisionedCertificates.length > 0;
|
|
17
17
|
}
|
|
18
18
|
catch (error) {
|
|
19
19
|
return false;
|
|
@@ -25,4 +25,4 @@ export class DeviceProvisionedCertificatesGuard {
|
|
|
25
25
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DeviceProvisionedCertificatesGuard, decorators: [{
|
|
26
26
|
type: Injectable
|
|
27
27
|
}], ctorParameters: () => [{ type: i1.DeviceProvisionedCertificatesService }] });
|
|
28
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
28
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGV2aWNlLXByb3Zpc2lvbmVkLWNlcnRpZmljYXRlcy5ndWFyZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL2RldmljZS1wcm92aXNpb25lZC1jZXJ0aWZpY2F0ZXMvZGV2aWNlLXByb3Zpc2lvbmVkLWNlcnRpZmljYXRlcy5ndWFyZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNDLE9BQU8sRUFBRSxvQ0FBb0MsRUFBRSxNQUFNLDJDQUEyQyxDQUFDOzs7QUFJakcsTUFBTSxPQUFPLGtDQUFrQztJQUM3QyxZQUFvQixvQ0FBMEU7UUFBMUUseUNBQW9DLEdBQXBDLG9DQUFvQyxDQUFzQztJQUFHLENBQUM7SUFFbEcsS0FBSyxDQUFDLFdBQVcsQ0FBQyxLQUE2QjtRQUM3QyxNQUFNLFdBQVcsR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLFdBQVcsSUFBSSxLQUFLLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUM7UUFDNUUsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1lBQ2pCLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQztRQUNELElBQUksQ0FBQztZQUNILE1BQU0sdUJBQXVCLEdBQzNCLE1BQU0sSUFBSSxDQUFDLG9DQUFvQyxDQUFDLGdDQUFnQyxDQUM5RSxXQUFXLENBQ1osQ0FBQztZQUNKLE9BQU8sdUJBQXVCLElBQUksdUJBQXVCLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztRQUN2RSxDQUFDO1FBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztZQUNmLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQztJQUNILENBQUM7K0dBakJVLGtDQUFrQzttSEFBbEMsa0NBQWtDOzs0RkFBbEMsa0NBQWtDO2tCQUQ5QyxVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRGV2aWNlUHJvdmlzaW9uZWRDZXJ0aWZpY2F0ZXNTZXJ2aWNlIH0gZnJvbSAnLi9kZXZpY2UtcHJvdmlzaW9uZWQtY2VydGlmaWNhdGVzLnNlcnZpY2UnO1xuaW1wb3J0IHsgQWN0aXZhdGVkUm91dGVTbmFwc2hvdCB9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBEZXZpY2VQcm92aXNpb25lZENlcnRpZmljYXRlc0d1YXJkIHtcbiAgY29uc3RydWN0b3IocHJpdmF0ZSBkZXZpY2VQcm92aXNpb25lZENlcnRpZmljYXRlc1NlcnZpY2U6IERldmljZVByb3Zpc2lvbmVkQ2VydGlmaWNhdGVzU2VydmljZSkge31cblxuICBhc3luYyBjYW5BY3RpdmF0ZShyb3V0ZTogQWN0aXZhdGVkUm91dGVTbmFwc2hvdCkge1xuICAgIGNvbnN0IGNvbnRleHREYXRhID0gcm91dGUuZGF0YS5jb250ZXh0RGF0YSB8fCByb3V0ZS5wYXJlbnQuZGF0YS5jb250ZXh0RGF0YTtcbiAgICBpZiAoIWNvbnRleHREYXRhKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIHRyeSB7XG4gICAgICBjb25zdCBwcm92aXNpb25lZENlcnRpZmljYXRlcyA9XG4gICAgICAgIGF3YWl0IHRoaXMuZGV2aWNlUHJvdmlzaW9uZWRDZXJ0aWZpY2F0ZXNTZXJ2aWNlLmdldERldmljZVByb3Zpc2lvbmVkQ2VydGlmaWNhdGVzKFxuICAgICAgICAgIGNvbnRleHREYXRhXG4gICAgICAgICk7XG4gICAgICByZXR1cm4gcHJvdmlzaW9uZWRDZXJ0aWZpY2F0ZXMgJiYgcHJvdmlzaW9uZWRDZXJ0aWZpY2F0ZXMubGVuZ3RoID4gMDtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgfVxufVxuIl19
|
package/esm2022/device-provisioned-certificates/device-tab-provisioned-certificates.component.mjs
CHANGED
|
@@ -4,7 +4,7 @@ import { DeviceProvisionedCertificatesService } from './device-provisioned-certi
|
|
|
4
4
|
import { ActivatedRoute } from '@angular/router';
|
|
5
5
|
import { CrlService } from '@c8y/client';
|
|
6
6
|
import { BehaviorSubject } from 'rxjs';
|
|
7
|
-
import { switchMap, map, tap, finalize } from 'rxjs/operators';
|
|
7
|
+
import { switchMap, map, tap, finalize, shareReplay } from 'rxjs/operators';
|
|
8
8
|
import { TranslateService } from '@ngx-translate/core';
|
|
9
9
|
import * as i0 from "@angular/core";
|
|
10
10
|
import * as i1 from "@c8y/ngx-components";
|
|
@@ -25,7 +25,9 @@ export class DeviceTabProvisionedCertificatesComponent {
|
|
|
25
25
|
this.reload = new BehaviorSubject(null);
|
|
26
26
|
this.provisionedCertificates = this.reload.pipe(tap(() => {
|
|
27
27
|
this.reloading = true;
|
|
28
|
-
}), map(() => this.route.snapshot.parent.data.contextData), switchMap(device => this.deviceSerialsService.getDeviceProvisionedCertificates(device)),
|
|
28
|
+
}), map(() => this.route.snapshot.parent.data.contextData), switchMap(device => this.deviceSerialsService.getDeviceProvisionedCertificates(device)), tap(() => {
|
|
29
|
+
this.reloading = false;
|
|
30
|
+
}), shareReplay(1), finalize(() => {
|
|
29
31
|
this.reloading = false;
|
|
30
32
|
}));
|
|
31
33
|
this.PX_ACTIONS = PX_ACTIONS;
|
|
@@ -67,10 +69,10 @@ export class DeviceTabProvisionedCertificatesComponent {
|
|
|
67
69
|
return await this.modalService.confirm(gettext('Revoke provisioned certificate'), this.translateService.instant(gettext(`You are about to revoke provisioned certificate {{ serialNumber }}. Do you want to proceed?`), { serialNumber }), Status.DANGER, { ok: gettext('Revoke'), cancel: gettext('Cancel') });
|
|
68
70
|
}
|
|
69
71
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DeviceTabProvisionedCertificatesComponent, deps: [{ token: i1.AlertService }, { token: i1.ModalService }, { token: i2.DeviceProvisionedCertificatesService }, { token: i3.CrlService }, { token: i4.ActivatedRoute }, { token: i5.TranslateService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
70
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DeviceTabProvisionedCertificatesComponent, selector: "device-tab-serials-component", ngImport: i0, template: "<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n title=\"{{ 'Reload' | translate }}\"\n type=\"button\"\n (click)=\"reload.next()\"\n >\n <i\n c8yIcon=\"refresh\"\n [ngClass]=\"{ 'icon-spin': reloading }\"\n ></i>\n {{ 'Reload' | translate }}\n </button>\n</c8y-action-bar-item>\n\n<!-- TODO: Add link to the mentioned documentation when it's available -->\n<c8y-ui-empty-state\n [icon]=\"'certificate'\"\n [title]=\"'No provisioned certificates to display.' | translate\"\n
|
|
72
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DeviceTabProvisionedCertificatesComponent, selector: "device-tab-serials-component", ngImport: i0, template: "<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n title=\"{{ 'Reload' | translate }}\"\n type=\"button\"\n (click)=\"reload.next()\"\n >\n <i\n c8yIcon=\"refresh\"\n [ngClass]=\"{ 'icon-spin': reloading }\"\n ></i>\n {{ 'Reload' | translate }}\n </button>\n</c8y-action-bar-item>\n\n<!-- TODO: Add link to the mentioned documentation when it's available -->\n<c8y-ui-empty-state\n [icon]=\"'certificate'\"\n [title]=\"'No provisioned certificates to display.' | translate\"\n *ngIf=\"(provisionedCertificates | async)?.length === 0\"\n></c8y-ui-empty-state>\n\n<div\n [ngStyle]=\"{ width: '50%' }\"\n *ngIf=\"(provisionedCertificates | async)?.length > 0\"\n>\n <div class=\"bg-level-0\">\n <div class=\"card-header separator\">\n <div\n class=\"card-title\"\n translate\n >\n Provisioned certificates\n </div>\n </div>\n <div class=\"p-16\">\n <c8y-list-group class=\"m-b-24\">\n <div\n class=\"page-sticky-header hidden-xs c8y-list__item\"\n style=\"display: block !important\"\n >\n <div class=\"c8y-list__item__block\">\n <div class=\"c8y-list__item__body\">\n <div class=\"content-flex-60\">\n <div class=\"col-5\">\n <span\n class=\"text-truncate\"\n title=\" {{ 'Serial No.' | translate }} \"\n >\n {{ 'Serial No.' | translate }}\n </span>\n </div>\n <div class=\"col-6\">\n <span\n class=\"text-truncate\"\n title=\"{{ 'Expiration date' | translate }}\"\n >\n {{ 'Expiration date' | translate }}\n </span>\n </div>\n <div class=\"col-1\"></div>\n </div>\n </div>\n </div>\n </div>\n\n <c8y-li\n *ngFor=\"let provisionedCertificate of provisionedCertificates | async; let i = index\"\n >\n <c8y-li-body class=\"content-flex-60\">\n <div class=\"col-5\">\n <div\n class=\"text-truncate\"\n title=\"{{ 'Serial No.' | translate }}: {{ provisionedCertificate.serialNumber }}\"\n >\n <span\n class=\"text-label-small m-t-8 m-r-8 visible-xs-inline\"\n translate\n >\n Serial No.\n </span>\n {{ provisionedCertificate.serialNumber }}\n </div>\n </div>\n\n <div class=\"col-6\">\n <div\n class=\"text-truncate\"\n title=\"{{ 'Expiration date' | translate }}: {{\n provisionedCertificate.validTill | c8yDate\n }}\"\n >\n <span\n class=\"text-label-small m-t-8 m-r-8 visible-xs-inline\"\n translate\n >\n Expiration date\n </span>\n <small\n *ngIf=\"provisionedCertificate.validTill\"\n [ngClass]=\"highlightDependingOnExpirationStatus(provisionedCertificate)\"\n >\n <i\n class=\"m-r-4\"\n c8yIcon=\"calendar\"\n *ngIf=\"!highlightDependingOnExpirationStatus(provisionedCertificate)\"\n ></i>\n <i\n class=\"m-r-4\"\n c8yIcon=\"warning\"\n *ngIf=\"highlightDependingOnExpirationStatus(provisionedCertificate)\"\n ></i>\n <span>{{ provisionedCertificate.validTill | c8yDate }}</span>\n </small>\n </div>\n </div>\n <div class=\"col-1\">\n <button\n class=\"btn btn-dot btn-dot--danger\"\n title=\"{{ 'Revoke' | translate }}\"\n type=\"button\"\n (click)=\"revokeProvisionedCertificate(provisionedCertificate)\"\n >\n <i c8yIcon=\"delete\"></i>\n </button>\n </div>\n </c8y-li-body>\n </c8y-li>\n </c8y-list-group>\n </div>\n </div>\n</div>\n", dependencies: [{ kind: "component", type: i1.ActionBarItemComponent, selector: "c8y-action-bar-item", inputs: ["placement", "priority", "itemClass", "injector", "groupId", "inGroupPriority"] }, { kind: "component", type: i1.EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "directive", type: i1.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i1.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i6.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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: i6.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i1.ListGroupComponent, selector: "c8y-list-group" }, { kind: "component", type: i1.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i1.ListItemBodyComponent, selector: "c8y-list-item-body, c8y-li-body", inputs: ["body"] }, { kind: "pipe", type: i1.C8yTranslatePipe, name: "translate" }, { kind: "pipe", type: i6.AsyncPipe, name: "async" }, { kind: "pipe", type: i1.DatePipe, name: "c8yDate" }] }); }
|
|
71
73
|
}
|
|
72
74
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DeviceTabProvisionedCertificatesComponent, decorators: [{
|
|
73
75
|
type: Component,
|
|
74
|
-
args: [{ selector: 'device-tab-serials-component', template: "<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n title=\"{{ 'Reload' | translate }}\"\n type=\"button\"\n (click)=\"reload.next()\"\n >\n <i\n c8yIcon=\"refresh\"\n [ngClass]=\"{ 'icon-spin': reloading }\"\n ></i>\n {{ 'Reload' | translate }}\n </button>\n</c8y-action-bar-item>\n\n<!-- TODO: Add link to the mentioned documentation when it's available -->\n<c8y-ui-empty-state\n [icon]=\"'certificate'\"\n [title]=\"'No provisioned certificates to display.' | translate\"\n
|
|
76
|
+
args: [{ selector: 'device-tab-serials-component', template: "<c8y-action-bar-item [placement]=\"'right'\">\n <button\n class=\"btn btn-link\"\n title=\"{{ 'Reload' | translate }}\"\n type=\"button\"\n (click)=\"reload.next()\"\n >\n <i\n c8yIcon=\"refresh\"\n [ngClass]=\"{ 'icon-spin': reloading }\"\n ></i>\n {{ 'Reload' | translate }}\n </button>\n</c8y-action-bar-item>\n\n<!-- TODO: Add link to the mentioned documentation when it's available -->\n<c8y-ui-empty-state\n [icon]=\"'certificate'\"\n [title]=\"'No provisioned certificates to display.' | translate\"\n *ngIf=\"(provisionedCertificates | async)?.length === 0\"\n></c8y-ui-empty-state>\n\n<div\n [ngStyle]=\"{ width: '50%' }\"\n *ngIf=\"(provisionedCertificates | async)?.length > 0\"\n>\n <div class=\"bg-level-0\">\n <div class=\"card-header separator\">\n <div\n class=\"card-title\"\n translate\n >\n Provisioned certificates\n </div>\n </div>\n <div class=\"p-16\">\n <c8y-list-group class=\"m-b-24\">\n <div\n class=\"page-sticky-header hidden-xs c8y-list__item\"\n style=\"display: block !important\"\n >\n <div class=\"c8y-list__item__block\">\n <div class=\"c8y-list__item__body\">\n <div class=\"content-flex-60\">\n <div class=\"col-5\">\n <span\n class=\"text-truncate\"\n title=\" {{ 'Serial No.' | translate }} \"\n >\n {{ 'Serial No.' | translate }}\n </span>\n </div>\n <div class=\"col-6\">\n <span\n class=\"text-truncate\"\n title=\"{{ 'Expiration date' | translate }}\"\n >\n {{ 'Expiration date' | translate }}\n </span>\n </div>\n <div class=\"col-1\"></div>\n </div>\n </div>\n </div>\n </div>\n\n <c8y-li\n *ngFor=\"let provisionedCertificate of provisionedCertificates | async; let i = index\"\n >\n <c8y-li-body class=\"content-flex-60\">\n <div class=\"col-5\">\n <div\n class=\"text-truncate\"\n title=\"{{ 'Serial No.' | translate }}: {{ provisionedCertificate.serialNumber }}\"\n >\n <span\n class=\"text-label-small m-t-8 m-r-8 visible-xs-inline\"\n translate\n >\n Serial No.\n </span>\n {{ provisionedCertificate.serialNumber }}\n </div>\n </div>\n\n <div class=\"col-6\">\n <div\n class=\"text-truncate\"\n title=\"{{ 'Expiration date' | translate }}: {{\n provisionedCertificate.validTill | c8yDate\n }}\"\n >\n <span\n class=\"text-label-small m-t-8 m-r-8 visible-xs-inline\"\n translate\n >\n Expiration date\n </span>\n <small\n *ngIf=\"provisionedCertificate.validTill\"\n [ngClass]=\"highlightDependingOnExpirationStatus(provisionedCertificate)\"\n >\n <i\n class=\"m-r-4\"\n c8yIcon=\"calendar\"\n *ngIf=\"!highlightDependingOnExpirationStatus(provisionedCertificate)\"\n ></i>\n <i\n class=\"m-r-4\"\n c8yIcon=\"warning\"\n *ngIf=\"highlightDependingOnExpirationStatus(provisionedCertificate)\"\n ></i>\n <span>{{ provisionedCertificate.validTill | c8yDate }}</span>\n </small>\n </div>\n </div>\n <div class=\"col-1\">\n <button\n class=\"btn btn-dot btn-dot--danger\"\n title=\"{{ 'Revoke' | translate }}\"\n type=\"button\"\n (click)=\"revokeProvisionedCertificate(provisionedCertificate)\"\n >\n <i c8yIcon=\"delete\"></i>\n </button>\n </div>\n </c8y-li-body>\n </c8y-li>\n </c8y-list-group>\n </div>\n </div>\n</div>\n" }]
|
|
75
77
|
}], ctorParameters: () => [{ type: i1.AlertService }, { type: i1.ModalService }, { type: i2.DeviceProvisionedCertificatesService }, { type: i3.CrlService }, { type: i4.ActivatedRoute }, { type: i5.TranslateService }] });
|
|
76
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"device-tab-provisioned-certificates.component.js","sourceRoot":"","sources":["../../../device-provisioned-certificates/device-tab-provisioned-certificates.component.ts","../../../device-provisioned-certificates/device-tab-provisioned-certificates.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit } from '@angular/core';\nimport { AlertService, gettext, ModalService, PX_ACTIONS, Status } from '@c8y/ngx-components';\nimport { DeviceProvisionedCertificatesService } from './device-provisioned-certificates.service';\nimport { ActivatedRoute } from '@angular/router';\nimport { CrlService, IManagedObject } from '@c8y/client';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { switchMap, map, tap, finalize } from 'rxjs/operators';\nimport { TranslateService } from '@ngx-translate/core';\n\nexport type ProvisionedCertificate = { serialNumber: string; validTill: string };\n\n@Component({\n  selector: 'device-tab-serials-component',\n  templateUrl: './device-tab-provisioned-certificates.component.html'\n})\nexport class DeviceTabProvisionedCertificatesComponent implements OnInit {\n  reloading = false;\n  reload: BehaviorSubject<void> = new BehaviorSubject(null);\n  provisionedCertificates: Observable<ProvisionedCertificate[]> = this.reload.pipe(\n    tap(() => {\n      this.reloading = true;\n    }),\n    map(() => this.route.snapshot.parent.data.contextData as IManagedObject),\n    switchMap(device => this.deviceSerialsService.getDeviceProvisionedCertificates(device)),\n    finalize(() => {\n      this.reloading = false;\n    })\n  );\n\n  protected readonly PX_ACTIONS = PX_ACTIONS;\n\n  constructor(\n    private alertService: AlertService,\n    private modalService: ModalService,\n    private deviceSerialsService: DeviceProvisionedCertificatesService,\n    private crlService: CrlService,\n    private route: ActivatedRoute,\n    private translateService: TranslateService\n  ) {}\n\n  ngOnInit(): void {\n    this.reload.next();\n  }\n\n  highlightDependingOnExpirationStatus(item: ProvisionedCertificate) {\n    const warningOffset = 24 * 60 * 60 * 1000 * 90; // 90 days\n    const todayTimestamp = new Date().getTime();\n    const warningTimestamp = new Date().getTime() + warningOffset;\n    const notAfterTimestamp = new Date(item.validTill).getTime();\n    const expired = notAfterTimestamp < todayTimestamp;\n    const expiresInLessThan90Days = notAfterTimestamp < warningTimestamp;\n    if (expired) {\n      return 'text-danger';\n    }\n    if (expiresInLessThan90Days) {\n      return 'text-warning';\n    }\n    return '';\n  }\n\n  async revokeProvisionedCertificate(provisionedCertificate: ProvisionedCertificate) {\n    if (provisionedCertificate) {\n      try {\n        const serialNumberInHex = provisionedCertificate.serialNumber;\n        await this.confirmRevocation(serialNumberInHex);\n        await this.crlService.uploadCrls([{ serialNumberInHex }]);\n      } catch (error) {\n        this.alertService.addServerFailure(error);\n      } finally {\n        this.reload.next();\n      }\n    }\n  }\n\n  private async confirmRevocation(serialNumber: string) {\n    return await this.modalService.confirm(\n      gettext('Revoke provisioned certificate'),\n      this.translateService.instant(\n        gettext(\n          `You are about to revoke provisioned certificate {{ serialNumber }}. Do you want to proceed?`\n        ),\n        { serialNumber }\n      ),\n      Status.DANGER,\n      { ok: gettext('Revoke'), cancel: gettext('Cancel') }\n    );\n  }\n}\n","<c8y-action-bar-item [placement]=\"'right'\">\n  <button\n    class=\"btn btn-link\"\n    title=\"{{ 'Reload' | translate }}\"\n    type=\"button\"\n    (click)=\"reload.next()\"\n  >\n    <i\n      c8yIcon=\"refresh\"\n      [ngClass]=\"{ 'icon-spin': reloading }\"\n    ></i>\n    {{ 'Reload' | translate }}\n  </button>\n</c8y-action-bar-item>\n\n<!-- TODO: Add link to the mentioned documentation when it's available -->\n<c8y-ui-empty-state\n  [icon]=\"'certificate'\"\n  [title]=\"'No provisioned certificates to display.' | translate\"\n  [subtitle]=\"\n    'This device can no longer connect to the platform because the last certificate has been revoked. Reissuing the certificate is not possible. To restore access, the device must be re-enrolled. Please follow the re-enrollment process as outlined in the documentation.'\n      | translate\n  \"\n  *ngIf=\"(provisionedCertificates | async)?.length === 0\"\n></c8y-ui-empty-state>\n\n<div\n  [ngStyle]=\"{ width: '50%' }\"\n  *ngIf=\"(provisionedCertificates | async)?.length > 0\"\n>\n  <div class=\"bg-level-0\">\n    <div class=\"card-header separator\">\n      <div\n        class=\"card-title\"\n        translate\n      >\n        Provisioned certificates\n      </div>\n    </div>\n    <div class=\"p-16\">\n      <c8y-list-group class=\"m-b-24\">\n        <div\n          class=\"page-sticky-header hidden-xs c8y-list__item\"\n          style=\"display: block !important\"\n        >\n          <div class=\"c8y-list__item__block\">\n            <div class=\"c8y-list__item__body\">\n              <div class=\"content-flex-60\">\n                <div class=\"col-5\">\n                  <span\n                    class=\"text-truncate\"\n                    title=\" {{ 'Serial No.' | translate }} \"\n                  >\n                    {{ 'Serial No.' | translate }}\n                  </span>\n                </div>\n                <div class=\"col-6\">\n                  <span\n                    class=\"text-truncate\"\n                    title=\"{{ 'Expiration date' | translate }}\"\n                  >\n                    {{ 'Expiration date' | translate }}\n                  </span>\n                </div>\n                <div class=\"col-1\"></div>\n              </div>\n            </div>\n          </div>\n        </div>\n\n        <c8y-li\n          *ngFor=\"let provisionedCertificate of provisionedCertificates | async; let i = index\"\n        >\n          <c8y-li-body class=\"content-flex-60\">\n            <div class=\"col-5\">\n              <div\n                class=\"text-truncate\"\n                title=\"{{ 'Serial No.' | translate }}: {{ provisionedCertificate.serialNumber }}\"\n              >\n                <span\n                  class=\"text-label-small m-t-8 m-r-8 visible-xs-inline\"\n                  translate\n                >\n                  Serial No.\n                </span>\n                {{ provisionedCertificate.serialNumber }}\n              </div>\n            </div>\n\n            <div class=\"col-6\">\n              <div\n                class=\"text-truncate\"\n                title=\"{{ 'Expiration date' | translate }}: {{\n                  provisionedCertificate.validTill | c8yDate\n                }}\"\n              >\n                <span\n                  class=\"text-label-small m-t-8 m-r-8 visible-xs-inline\"\n                  translate\n                >\n                  Expiration date\n                </span>\n                <small\n                  *ngIf=\"provisionedCertificate.validTill\"\n                  [ngClass]=\"highlightDependingOnExpirationStatus(provisionedCertificate)\"\n                >\n                  <i\n                    class=\"m-r-4\"\n                    c8yIcon=\"calendar\"\n                    *ngIf=\"!highlightDependingOnExpirationStatus(provisionedCertificate)\"\n                  ></i>\n                  <i\n                    class=\"m-r-4\"\n                    c8yIcon=\"warning\"\n                    *ngIf=\"highlightDependingOnExpirationStatus(provisionedCertificate)\"\n                  ></i>\n                  <span>{{ provisionedCertificate.validTill | c8yDate }}</span>\n                </small>\n              </div>\n            </div>\n            <div class=\"col-1\">\n              <button\n                class=\"btn btn-dot btn-dot--danger\"\n                title=\"{{ 'Revoke' | translate }}\"\n                type=\"button\"\n                (click)=\"revokeProvisionedCertificate(provisionedCertificate)\"\n              >\n                <i c8yIcon=\"delete\"></i>\n              </button>\n            </div>\n          </c8y-li-body>\n        </c8y-li>\n      </c8y-list-group>\n    </div>\n  </div>\n</div>\n"]}
|
|
78
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"device-tab-provisioned-certificates.component.js","sourceRoot":"","sources":["../../../device-provisioned-certificates/device-tab-provisioned-certificates.component.ts","../../../device-provisioned-certificates/device-tab-provisioned-certificates.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAU,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,YAAY,EAAE,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,qBAAqB,CAAC;AAC9F,OAAO,EAAE,oCAAoC,EAAE,MAAM,2CAA2C,CAAC;AACjG,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,UAAU,EAAkB,MAAM,aAAa,CAAC;AACzD,OAAO,EAAE,eAAe,EAAc,MAAM,MAAM,CAAC;AACnD,OAAO,EAAE,SAAS,EAAE,GAAG,EAAE,GAAG,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC5E,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;;;;;;;;AAQvD,MAAM,OAAO,yCAAyC;IAoBpD,YACU,YAA0B,EAC1B,YAA0B,EAC1B,oBAA0D,EAC1D,UAAsB,EACtB,KAAqB,EACrB,gBAAkC;QALlC,iBAAY,GAAZ,YAAY,CAAc;QAC1B,iBAAY,GAAZ,YAAY,CAAc;QAC1B,yBAAoB,GAApB,oBAAoB,CAAsC;QAC1D,eAAU,GAAV,UAAU,CAAY;QACtB,UAAK,GAAL,KAAK,CAAgB;QACrB,qBAAgB,GAAhB,gBAAgB,CAAkB;QAzB5C,cAAS,GAAG,KAAK,CAAC;QAClB,WAAM,GAA0B,IAAI,eAAe,CAAC,IAAI,CAAC,CAAC;QAC1D,4BAAuB,GAAyC,IAAI,CAAC,MAAM,CAAC,IAAI,CAC9E,GAAG,CAAC,GAAG,EAAE;YACP,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACxB,CAAC,CAAC,EACF,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,WAA6B,CAAC,EACxE,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,gCAAgC,CAAC,MAAM,CAAC,CAAC,EACvF,GAAG,CAAC,GAAG,EAAE;YACP,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACzB,CAAC,CAAC,EACF,WAAW,CAAC,CAAC,CAAC,EACd,QAAQ,CAAC,GAAG,EAAE;YACZ,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACzB,CAAC,CAAC,CACH,CAAC;QAEiB,eAAU,GAAG,UAAU,CAAC;IASxC,CAAC;IAEJ,QAAQ;QACN,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;IACrB,CAAC;IAED,oCAAoC,CAAC,IAA4B;QAC/D,MAAM,aAAa,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,GAAG,EAAE,CAAC,CAAC,UAAU;QAC1D,MAAM,cAAc,GAAG,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,CAAC;QAC5C,MAAM,gBAAgB,GAAG,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,GAAG,aAAa,CAAC;QAC9D,MAAM,iBAAiB,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,OAAO,EAAE,CAAC;QAC7D,MAAM,OAAO,GAAG,iBAAiB,GAAG,cAAc,CAAC;QACnD,MAAM,uBAAuB,GAAG,iBAAiB,GAAG,gBAAgB,CAAC;QACrE,IAAI,OAAO,EAAE,CAAC;YACZ,OAAO,aAAa,CAAC;QACvB,CAAC;QACD,IAAI,uBAAuB,EAAE,CAAC;YAC5B,OAAO,cAAc,CAAC;QACxB,CAAC;QACD,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,KAAK,CAAC,4BAA4B,CAAC,sBAA8C;QAC/E,IAAI,sBAAsB,EAAE,CAAC;YAC3B,IAAI,CAAC;gBACH,MAAM,iBAAiB,GAAG,sBAAsB,CAAC,YAAY,CAAC;gBAC9D,MAAM,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;gBAChD,MAAM,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,EAAE,iBAAiB,EAAE,CAAC,CAAC,CAAC;YAC5D,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,IAAI,CAAC,YAAY,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;YAC5C,CAAC;oBAAS,CAAC;gBACT,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;YACrB,CAAC;QACH,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,iBAAiB,CAAC,YAAoB;QAClD,OAAO,MAAM,IAAI,CAAC,YAAY,CAAC,OAAO,CACpC,OAAO,CAAC,gCAAgC,CAAC,EACzC,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAC3B,OAAO,CACL,6FAA6F,CAC9F,EACD,EAAE,YAAY,EAAE,CACjB,EACD,MAAM,CAAC,MAAM,EACb,EAAE,EAAE,EAAE,OAAO,CAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,QAAQ,CAAC,EAAE,CACrD,CAAC;IACJ,CAAC;+GA3EU,yCAAyC;mGAAzC,yCAAyC,oECftD,m2IAoIA;;4FDrHa,yCAAyC;kBAJrD,SAAS;+BACE,8BAA8B","sourcesContent":["import { Component, OnInit } from '@angular/core';\nimport { AlertService, gettext, ModalService, PX_ACTIONS, Status } from '@c8y/ngx-components';\nimport { DeviceProvisionedCertificatesService } from './device-provisioned-certificates.service';\nimport { ActivatedRoute } from '@angular/router';\nimport { CrlService, IManagedObject } from '@c8y/client';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { switchMap, map, tap, finalize, shareReplay } from 'rxjs/operators';\nimport { TranslateService } from '@ngx-translate/core';\n\nexport type ProvisionedCertificate = { serialNumber: string; validTill: string };\n\n@Component({\n  selector: 'device-tab-serials-component',\n  templateUrl: './device-tab-provisioned-certificates.component.html'\n})\nexport class DeviceTabProvisionedCertificatesComponent implements OnInit {\n  reloading = false;\n  reload: BehaviorSubject<void> = new BehaviorSubject(null);\n  provisionedCertificates: Observable<ProvisionedCertificate[]> = this.reload.pipe(\n    tap(() => {\n      this.reloading = true;\n    }),\n    map(() => this.route.snapshot.parent.data.contextData as IManagedObject),\n    switchMap(device => this.deviceSerialsService.getDeviceProvisionedCertificates(device)),\n    tap(() => {\n      this.reloading = false;\n    }),\n    shareReplay(1),\n    finalize(() => {\n      this.reloading = false;\n    })\n  );\n\n  protected readonly PX_ACTIONS = PX_ACTIONS;\n\n  constructor(\n    private alertService: AlertService,\n    private modalService: ModalService,\n    private deviceSerialsService: DeviceProvisionedCertificatesService,\n    private crlService: CrlService,\n    private route: ActivatedRoute,\n    private translateService: TranslateService\n  ) {}\n\n  ngOnInit(): void {\n    this.reload.next();\n  }\n\n  highlightDependingOnExpirationStatus(item: ProvisionedCertificate) {\n    const warningOffset = 24 * 60 * 60 * 1000 * 90; // 90 days\n    const todayTimestamp = new Date().getTime();\n    const warningTimestamp = new Date().getTime() + warningOffset;\n    const notAfterTimestamp = new Date(item.validTill).getTime();\n    const expired = notAfterTimestamp < todayTimestamp;\n    const expiresInLessThan90Days = notAfterTimestamp < warningTimestamp;\n    if (expired) {\n      return 'text-danger';\n    }\n    if (expiresInLessThan90Days) {\n      return 'text-warning';\n    }\n    return '';\n  }\n\n  async revokeProvisionedCertificate(provisionedCertificate: ProvisionedCertificate) {\n    if (provisionedCertificate) {\n      try {\n        const serialNumberInHex = provisionedCertificate.serialNumber;\n        await this.confirmRevocation(serialNumberInHex);\n        await this.crlService.uploadCrls([{ serialNumberInHex }]);\n      } catch (error) {\n        this.alertService.addServerFailure(error);\n      } finally {\n        this.reload.next();\n      }\n    }\n  }\n\n  private async confirmRevocation(serialNumber: string) {\n    return await this.modalService.confirm(\n      gettext('Revoke provisioned certificate'),\n      this.translateService.instant(\n        gettext(\n          `You are about to revoke provisioned certificate {{ serialNumber }}. Do you want to proceed?`\n        ),\n        { serialNumber }\n      ),\n      Status.DANGER,\n      { ok: gettext('Revoke'), cancel: gettext('Cancel') }\n    );\n  }\n}\n","<c8y-action-bar-item [placement]=\"'right'\">\n  <button\n    class=\"btn btn-link\"\n    title=\"{{ 'Reload' | translate }}\"\n    type=\"button\"\n    (click)=\"reload.next()\"\n  >\n    <i\n      c8yIcon=\"refresh\"\n      [ngClass]=\"{ 'icon-spin': reloading }\"\n    ></i>\n    {{ 'Reload' | translate }}\n  </button>\n</c8y-action-bar-item>\n\n<!-- TODO: Add link to the mentioned documentation when it's available -->\n<c8y-ui-empty-state\n  [icon]=\"'certificate'\"\n  [title]=\"'No provisioned certificates to display.' | translate\"\n  *ngIf=\"(provisionedCertificates | async)?.length === 0\"\n></c8y-ui-empty-state>\n\n<div\n  [ngStyle]=\"{ width: '50%' }\"\n  *ngIf=\"(provisionedCertificates | async)?.length > 0\"\n>\n  <div class=\"bg-level-0\">\n    <div class=\"card-header separator\">\n      <div\n        class=\"card-title\"\n        translate\n      >\n        Provisioned certificates\n      </div>\n    </div>\n    <div class=\"p-16\">\n      <c8y-list-group class=\"m-b-24\">\n        <div\n          class=\"page-sticky-header hidden-xs c8y-list__item\"\n          style=\"display: block !important\"\n        >\n          <div class=\"c8y-list__item__block\">\n            <div class=\"c8y-list__item__body\">\n              <div class=\"content-flex-60\">\n                <div class=\"col-5\">\n                  <span\n                    class=\"text-truncate\"\n                    title=\" {{ 'Serial No.' | translate }} \"\n                  >\n                    {{ 'Serial No.' | translate }}\n                  </span>\n                </div>\n                <div class=\"col-6\">\n                  <span\n                    class=\"text-truncate\"\n                    title=\"{{ 'Expiration date' | translate }}\"\n                  >\n                    {{ 'Expiration date' | translate }}\n                  </span>\n                </div>\n                <div class=\"col-1\"></div>\n              </div>\n            </div>\n          </div>\n        </div>\n\n        <c8y-li\n          *ngFor=\"let provisionedCertificate of provisionedCertificates | async; let i = index\"\n        >\n          <c8y-li-body class=\"content-flex-60\">\n            <div class=\"col-5\">\n              <div\n                class=\"text-truncate\"\n                title=\"{{ 'Serial No.' | translate }}: {{ provisionedCertificate.serialNumber }}\"\n              >\n                <span\n                  class=\"text-label-small m-t-8 m-r-8 visible-xs-inline\"\n                  translate\n                >\n                  Serial No.\n                </span>\n                {{ provisionedCertificate.serialNumber }}\n              </div>\n            </div>\n\n            <div class=\"col-6\">\n              <div\n                class=\"text-truncate\"\n                title=\"{{ 'Expiration date' | translate }}: {{\n                  provisionedCertificate.validTill | c8yDate\n                }}\"\n              >\n                <span\n                  class=\"text-label-small m-t-8 m-r-8 visible-xs-inline\"\n                  translate\n                >\n                  Expiration date\n                </span>\n                <small\n                  *ngIf=\"provisionedCertificate.validTill\"\n                  [ngClass]=\"highlightDependingOnExpirationStatus(provisionedCertificate)\"\n                >\n                  <i\n                    class=\"m-r-4\"\n                    c8yIcon=\"calendar\"\n                    *ngIf=\"!highlightDependingOnExpirationStatus(provisionedCertificate)\"\n                  ></i>\n                  <i\n                    class=\"m-r-4\"\n                    c8yIcon=\"warning\"\n                    *ngIf=\"highlightDependingOnExpirationStatus(provisionedCertificate)\"\n                  ></i>\n                  <span>{{ provisionedCertificate.validTill | c8yDate }}</span>\n                </small>\n              </div>\n            </div>\n            <div class=\"col-1\">\n              <button\n                class=\"btn btn-dot btn-dot--danger\"\n                title=\"{{ 'Revoke' | translate }}\"\n                type=\"button\"\n                (click)=\"revokeProvisionedCertificate(provisionedCertificate)\"\n              >\n                <i c8yIcon=\"delete\"></i>\n              </button>\n            </div>\n          </c8y-li-body>\n        </c8y-li>\n      </c8y-list-group>\n    </div>\n  </div>\n</div>\n"]}
|