@c8y/ngx-components 1021.71.1 → 1021.71.2

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.
@@ -1 +1 @@
1
- {"version":3,"file":"beta-preview-drawer.factory.d.ts","sourceRoot":"","sources":["../../../core/beta-feature/beta-preview-drawer.factory.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAC7D,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AACpD,OAAO,EAAiB,UAAU,EAAE,MAAM,MAAM,CAAC;AAGjD,OAAO,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AACxD,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;;AAE5D,qBACa,wBAAyB,YAAW,gBAAgB,CAAC,UAAU,CAAC;IASzE,OAAO,CAAC,OAAO;IACf,OAAO,CAAC,cAAc;IACtB,OAAO,CAAC,WAAW;IAVrB,SAAS,CAAC,UAAU,EAAE,UAAU,CAK9B;gBAGQ,OAAO,EAAE,cAAc,EACvB,cAAc,EAAE,cAAc,EAC9B,WAAW,EAAE,WAAW;IAGlC,GAAG,IAAI,UAAU,CAAC,UAAU,EAAE,GAAG,UAAU,CAAC;IAY5C,OAAO,CAAC,sBAAsB;yCA1BnB,wBAAwB;6CAAxB,wBAAwB;CA4CpC"}
1
+ {"version":3,"file":"beta-preview-drawer.factory.d.ts","sourceRoot":"","sources":["../../../core/beta-feature/beta-preview-drawer.factory.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAC7D,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AACpD,OAAO,EAAiB,UAAU,EAAE,MAAM,MAAM,CAAC;AAGjD,OAAO,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AACxD,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;;AAE5D,qBACa,wBAAyB,YAAW,gBAAgB,CAAC,UAAU,CAAC;IASzE,OAAO,CAAC,OAAO;IACf,OAAO,CAAC,cAAc;IACtB,OAAO,CAAC,WAAW;IAVrB,SAAS,CAAC,UAAU,EAAE,UAAU,CAK9B;gBAGQ,OAAO,EAAE,cAAc,EACvB,cAAc,EAAE,cAAc,EAC9B,WAAW,EAAE,WAAW;IAGlC,GAAG,IAAI,UAAU,CAAC,UAAU,EAAE,GAAG,UAAU,CAAC;IAY5C,OAAO,CAAC,sBAAsB;yCA1BnB,wBAAwB;6CAAxB,wBAAwB;CA+CpC"}
@@ -17,7 +17,6 @@ export declare class BetaPreviewComponent {
17
17
  updateDescription(): Promise<void>;
18
18
  isComponent(description: string | Type<any> | Promise<string | Type<any>>): description is Type<any>;
19
19
  onToggle(feature: BetaFeature, event: Event): Promise<void>;
20
- featureKeyExists(key: string): Observable<boolean>;
21
20
  close(): void;
22
21
  static ɵfac: i0.ɵɵFactoryDeclaration<BetaPreviewComponent, never>;
23
22
  static ɵcmp: i0.ɵɵComponentDeclaration<BetaPreviewComponent, "c8y-beta-preview", never, {}, {}, never, never, true, never>;
@@ -1 +1 @@
1
- {"version":3,"file":"beta-preview.component.d.ts","sourceRoot":"","sources":["../../../core/beta-feature/beta-preview.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAqB,IAAI,EAAE,MAAM,eAAe,CAAC;AACxD,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAE,eAAe,EAAkB,UAAU,EAAE,MAAM,MAAM,CAAC;;AAQnE,qBAQa,oBAAoB;IAC/B,eAAe,EAAE,WAAW,CAAC;IAC7B,WAAW,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC;IAC9B,YAAY,EAAE,eAAe,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CACrD;IAC1B,OAAO,CAAC,YAAY,CAAmD;IAEvE,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAA0B;IAC7D,OAAO,CAAC,QAAQ,CAAC,mBAAmB,CAA+B;IACnE,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAA8B;IAEjE,SAAS,EAAE,UAAU,CAAC,WAAW,EAAE,CAAC,CAAkC;;IAQhE,QAAQ;IAMR,qBAAqB,CAAC,OAAO,EAAE,WAAW;IAK1C,iBAAiB;IAUvB,WAAW,CAAC,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC;IAInE,QAAQ,CAAC,OAAO,EAAE,WAAW,EAAE,KAAK,EAAE,KAAK,GAAG,OAAO,CAAC,IAAI,CAAC;IAKjE,gBAAgB,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC;IAIlD,KAAK;yCArDM,oBAAoB;2CAApB,oBAAoB;CAwDhC"}
1
+ {"version":3,"file":"beta-preview.component.d.ts","sourceRoot":"","sources":["../../../core/beta-feature/beta-preview.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAqB,IAAI,EAAE,MAAM,eAAe,CAAC;AACxD,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAE,eAAe,EAAkB,UAAU,EAAE,MAAM,MAAM,CAAC;;AAQnE,qBAQa,oBAAoB;IAC/B,eAAe,EAAE,WAAW,CAAC;IAC7B,WAAW,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC;IAC9B,YAAY,EAAE,eAAe,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CACrD;IAC1B,OAAO,CAAC,YAAY,CAAmD;IAEvE,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAA0B;IAC7D,OAAO,CAAC,QAAQ,CAAC,mBAAmB,CAA+B;IACnE,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAA8B;IAEjE,SAAS,EAAE,UAAU,CAAC,WAAW,EAAE,CAAC,CAAmD;;IAQjF,QAAQ;IAMR,qBAAqB,CAAC,OAAO,EAAE,WAAW;IAK1C,iBAAiB;IAUvB,WAAW,CAAC,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC;IAInE,QAAQ,CAAC,OAAO,EAAE,WAAW,EAAE,KAAK,EAAE,KAAK,GAAG,OAAO,CAAC,IAAI,CAAC;IAKjE,KAAK;yCAjDM,oBAAoB;2CAApB,oBAAoB;CAoDhC"}
@@ -35,6 +35,10 @@ export declare class PreviewService extends ExtensionPointForPlugins<BetaFeature
35
35
  * This assumes all features are provided via the `items$` observable.
36
36
  */
37
37
  markAllFeaturesAsSeen(): Promise<void>;
38
+ /**
39
+ * Filters the features to only include those that are available by checking the feature toggle API.
40
+ */
41
+ getAvailableFeatures$(): Observable<BetaFeature[]>;
38
42
  /**
39
43
  *
40
44
  * @returns An array of feature names that have been seen by the user. This is retrieved from local storage.
@@ -1 +1 @@
1
- {"version":3,"file":"beta-preview.service.d.ts","sourceRoot":"","sources":["../../../core/beta-feature/beta-preview.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAsB,cAAc,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAC7E,OAAO,EAAE,eAAe,EAAiC,UAAU,EAAM,MAAM,MAAM,CAAC;AAEtF,OAAO,EACL,kBAAkB,EAClB,eAAe,EAGf,wBAAwB,EAIzB,MAAM,2BAA2B,CAAC;AACnC,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAE,qBAAqB,EAAE,MAAM,oCAAoC,CAAC;;AAK3E,eAAO,MAAM,YAAY,+BAAoD,CAAC;AAE9E,wBAAgB,WAAW,CACzB,OAAO,EAAE,eAAe,CAAC,WAAW,CAAC,EACrC,OAAO,CAAC,EAAE,OAAO,CAAC,kBAAkB,CAAC,4HAGtC;AAED,qBACa,cAAe,SAAQ,wBAAwB,CAAC,WAAW,CAAC;IACvE,MAAM,EAAE,UAAU,CAAC,WAAW,EAAE,CAAC,CAA+B;IAChE,OAAO,EAAE,eAAe,CAAC,OAAO,CAAC,CAAuC;IACxE,OAAO,CAAC,QAAQ,CAAC,iBAAiB,CAA4B;IAC9D,OAAO,CAAC,aAAa,CAAyD;IAC9E,kBAAkB,sBAGhB;IACF,OAAO,CAAC,QAAQ,CAAC,cAAc,CAA0B;IACzD,OAAO,CAAC,QAAQ,CAAC,mBAAmB,CAA+B;IACnE,OAAO,CAAC,QAAQ,CAAC,cAAc,CAA0B;IAEzD,OAAO,CAAC,yBAAyB,CAA0B;gBAE/C,YAAY,EAAE,QAAQ,EAAE,aAAa,EAAE,qBAAqB;IAIxE,IAAI,KAAK,IAAI,GAAG,CAAC,WAAW,CAAC,CAE5B;IAED;;;;OAIG;IACH,SAAS,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC;IAuB3C;;;;OAIG;IACG,qBAAqB,CAAC,GAAG,EAAE,MAAM,EAAE,WAAW,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC;IA+B7E;;;OAGG;IACG,qBAAqB,IAAI,OAAO,CAAC,IAAI,CAAC;IAQ5C;;;OAGG;IACH,OAAO,CAAC,eAAe;IAKvB,SAAS,CAAC,oBAAoB,IAAI,UAAU,CAAC,WAAW,EAAE,CAAC;yCA5GhD,cAAc;6CAAd,cAAc;CAuH1B"}
1
+ {"version":3,"file":"beta-preview.service.d.ts","sourceRoot":"","sources":["../../../core/beta-feature/beta-preview.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAsB,cAAc,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAC7E,OAAO,EAAE,eAAe,EAAiC,UAAU,EAAM,MAAM,MAAM,CAAC;AAEtF,OAAO,EACL,kBAAkB,EAClB,eAAe,EAGf,wBAAwB,EAIzB,MAAM,2BAA2B,CAAC;AACnC,OAAO,EAAE,WAAW,EAAE,MAAM,sBAAsB,CAAC;AACnD,OAAO,EAAE,qBAAqB,EAAE,MAAM,oCAAoC,CAAC;;AAK3E,eAAO,MAAM,YAAY,+BAAoD,CAAC;AAE9E,wBAAgB,WAAW,CACzB,OAAO,EAAE,eAAe,CAAC,WAAW,CAAC,EACrC,OAAO,CAAC,EAAE,OAAO,CAAC,kBAAkB,CAAC,4HAGtC;AAED,qBACa,cAAe,SAAQ,wBAAwB,CAAC,WAAW,CAAC;IACvE,MAAM,EAAE,UAAU,CAAC,WAAW,EAAE,CAAC,CAA+B;IAChE,OAAO,EAAE,eAAe,CAAC,OAAO,CAAC,CAAuC;IACxE,OAAO,CAAC,QAAQ,CAAC,iBAAiB,CAA4B;IAC9D,OAAO,CAAC,aAAa,CAAyD;IAC9E,kBAAkB,sBAGhB;IACF,OAAO,CAAC,QAAQ,CAAC,cAAc,CAA0B;IACzD,OAAO,CAAC,QAAQ,CAAC,mBAAmB,CAA+B;IACnE,OAAO,CAAC,QAAQ,CAAC,cAAc,CAA0B;IAEzD,OAAO,CAAC,yBAAyB,CAA0B;gBAE/C,YAAY,EAAE,QAAQ,EAAE,aAAa,EAAE,qBAAqB;IAIxE,IAAI,KAAK,IAAI,GAAG,CAAC,WAAW,CAAC,CAE5B;IAED;;;;OAIG;IACH,SAAS,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC;IAuB3C;;;;OAIG;IACG,qBAAqB,CAAC,GAAG,EAAE,MAAM,EAAE,WAAW,EAAE,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC;IA+B7E;;;OAGG;IACG,qBAAqB,IAAI,OAAO,CAAC,IAAI,CAAC;IAQ5C;;OAEG;IACH,qBAAqB,IAAI,UAAU,CAAC,WAAW,EAAE,CAAC;IAgBlD;;;OAGG;IACH,OAAO,CAAC,eAAe;IAKvB,SAAS,CAAC,oBAAoB,IAAI,UAAU,CAAC,WAAW,EAAE,CAAC;yCA/HhD,cAAc;6CAAd,cAAc;CA0I1B"}
@@ -1 +1 @@
1
- {"version":3,"file":"feature-cache.service.d.ts","sourceRoot":"","sources":["../../../core/beta-feature/feature-cache.service.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,cAAc,EAAkB,MAAM,aAAa,CAAC;AAC7D,OAAO,EAAmC,UAAU,EAAE,MAAM,MAAM,CAAC;;AAEnE,qBACa,mBAAmB;IAMlB,OAAO,CAAC,cAAc;IALlC,OAAO,CAAC,cAAc,CAAwC;IAC9D,OAAO,CAAC,SAAS,CAGf;gBACkB,cAAc,EAAE,cAAc;IAElD;;;OAGG;IACH,sBAAsB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAIrD;;;;OAIG;IACH,aAAa,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC;IAI/C,eAAe,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC;IASjD,iBAAiB,IAAI,IAAI;YAIX,YAAY;yCAtCf,mBAAmB;6CAAnB,mBAAmB;CA0C/B"}
1
+ {"version":3,"file":"feature-cache.service.d.ts","sourceRoot":"","sources":["../../../core/beta-feature/feature-cache.service.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,cAAc,EAAkB,MAAM,aAAa,CAAC;AAC7D,OAAO,EAAmC,UAAU,EAAE,MAAM,MAAM,CAAC;;AAEnE,qBACa,mBAAmB;IAMlB,OAAO,CAAC,cAAc;IALlC,OAAO,CAAC,cAAc,CAAwC;IAC9D,OAAO,CAAC,SAAS,CAGf;gBACkB,cAAc,EAAE,cAAc;IAElD;;;OAGG;IACH,sBAAsB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAIrD;;;;OAIG;IACH,aAAa,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC;IAI/C,eAAe,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC;IASjD,iBAAiB,IAAI,IAAI;YAIX,YAAY;yCAtCf,mBAAmB;6CAAnB,mBAAmB;CA+C/B"}
@@ -30,7 +30,10 @@ export class BetaPreviewDrawerFactory {
30
30
  }));
31
31
  }
32
32
  shouldShowBetaPreview$() {
33
- return combineLatest([this.options.get$('hideBetaPreview'), this.previewService.items$]).pipe(map(([hideBetaPreview, items]) => {
33
+ return combineLatest([
34
+ this.options.get$('hideBetaPreview'),
35
+ this.previewService.getAvailableFeatures$()
36
+ ]).pipe(map(([hideBetaPreview, items]) => {
34
37
  const hasItems = items.length > 0;
35
38
  if (!hasItems) {
36
39
  return false;
@@ -49,4 +52,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
49
52
  type: Injectable,
50
53
  args: [{ providedIn: 'root' }]
51
54
  }], ctorParameters: () => [{ type: i1.OptionsService }, { type: i2.PreviewService }, { type: i3.Permissions }] });
52
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmV0YS1wcmV2aWV3LWRyYXdlci5mYWN0b3J5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vY29yZS9iZXRhLWZlYXR1cmUvYmV0YS1wcmV2aWV3LWRyYXdlci5mYWN0b3J5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFM0MsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBRTNELE9BQU8sRUFBRSxhQUFhLEVBQWMsTUFBTSxNQUFNLENBQUM7QUFDakQsT0FBTyxFQUFFLG9CQUFvQixFQUFFLEdBQUcsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzNELE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQzdFLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUN4RCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sK0JBQStCLENBQUM7Ozs7O0FBRzVELE1BQU0sT0FBTyx3QkFBd0I7SUFRbkMsWUFDVSxPQUF1QixFQUN2QixjQUE4QixFQUM5QixXQUF3QjtRQUZ4QixZQUFPLEdBQVAsT0FBTyxDQUFnQjtRQUN2QixtQkFBYyxHQUFkLGNBQWMsQ0FBZ0I7UUFDOUIsZ0JBQVcsR0FBWCxXQUFXLENBQWE7UUFWeEIsZUFBVSxHQUFlO1lBQ2pDLFNBQVMsRUFBRSwwQkFBMEI7WUFDckMsUUFBUSxFQUFFLE9BQU87WUFDakIsUUFBUSxFQUFFLEVBQUU7WUFDWixFQUFFLEVBQUUsYUFBYTtTQUNsQixDQUFDO0lBTUMsQ0FBQztJQUVKLEdBQUc7UUFDRCxPQUFPLElBQUksQ0FBQyxzQkFBc0IsRUFBRSxDQUFDLElBQUksQ0FDdkMsb0JBQW9CLEVBQUUsRUFDdEIsR0FBRyxDQUFDLHFCQUFxQixDQUFDLEVBQUU7WUFDMUIsSUFBSSxxQkFBcUIsRUFBRSxDQUFDO2dCQUMxQixPQUFPLElBQUksQ0FBQyxVQUFVLENBQUM7WUFDekIsQ0FBQztZQUNELE9BQU8sRUFBRSxDQUFDO1FBQ1osQ0FBQyxDQUFDLENBQ0gsQ0FBQztJQUNKLENBQUM7SUFFTyxzQkFBc0I7UUFDNUIsT0FBTyxhQUFhLENBQUMsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxFQUFFLElBQUksQ0FBQyxjQUFjLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQzNGLEdBQUcsQ0FBQyxDQUFDLENBQUMsZUFBZSxFQUFFLEtBQUssQ0FBQyxFQUFFLEVBQUU7WUFDL0IsTUFBTSxRQUFRLEdBQUcsS0FBSyxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUM7WUFDbEMsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO2dCQUNkLE9BQU8sS0FBSyxDQUFDO1lBQ2YsQ0FBQztZQUVELE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsT0FBTyxDQUFDLFdBQVcsQ0FBQyw0QkFBNEIsQ0FBQyxDQUFDO1lBRXhGLElBQUksZUFBZSxLQUFLLFNBQVMsRUFBRSxDQUFDO2dCQUNsQyxPQUFPLFFBQVEsSUFBSSxZQUFZLENBQUM7WUFDbEMsQ0FBQztZQUVELE9BQU8sQ0FBQyxlQUFlLElBQUksUUFBUSxJQUFJLFlBQVksQ0FBQztRQUN0RCxDQUFDLENBQUMsQ0FDSCxDQUFDO0lBQ0osQ0FBQzsrR0EzQ1Usd0JBQXdCO21IQUF4Qix3QkFBd0IsY0FEWCxNQUFNOzs0RkFDbkIsd0JBQXdCO2tCQURwQyxVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEV4dGVuc2lvbkZhY3RvcnkgfSBmcm9tICcuLi9jb21tb24vZXh0ZW5zaW9uLWhvb2tzJztcbmltcG9ydCB7IE9wdGlvbnNTZXJ2aWNlIH0gZnJvbSAnLi4vY29tbW9uL29wdGlvbnMuc2VydmljZSc7XG5pbXBvcnQgeyBEcmF3ZXJJdGVtIH0gZnJvbSAnLi4vZHJhd2VyL2RyYXdlci5tb2RlbCc7XG5pbXBvcnQgeyBjb21iaW5lTGF0ZXN0LCBPYnNlcnZhYmxlIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBkaXN0aW5jdFVudGlsQ2hhbmdlZCwgbWFwIH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHsgQmV0YVByZXZpZXdCdXR0b25Db21wb25lbnQgfSBmcm9tICcuL2JldGEtcHJldmlldy1idXR0b24uY29tcG9uZW50JztcbmltcG9ydCB7IFByZXZpZXdTZXJ2aWNlIH0gZnJvbSAnLi9iZXRhLXByZXZpZXcuc2VydmljZSc7XG5pbXBvcnQgeyBQZXJtaXNzaW9ucyB9IGZyb20gJy4uL2NvbW1vbi9wZXJtaXNzaW9ucy5zZXJ2aWNlJztcblxuQEluamVjdGFibGUoeyBwcm92aWRlZEluOiAncm9vdCcgfSlcbmV4cG9ydCBjbGFzcyBCZXRhUHJldmlld0RyYXdlckZhY3RvcnkgaW1wbGVtZW50cyBFeHRlbnNpb25GYWN0b3J5PERyYXdlckl0ZW0+IHtcbiAgcHJvdGVjdGVkIGRyYXdlckl0ZW06IERyYXdlckl0ZW0gPSB7XG4gICAgY29tcG9uZW50OiBCZXRhUHJldmlld0J1dHRvbkNvbXBvbmVudCxcbiAgICBwb3NpdGlvbjogJ3JpZ2h0JyxcbiAgICBwcmlvcml0eTogODAsXG4gICAgaWQ6ICdiZXRhUHJldmlldydcbiAgfTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIG9wdGlvbnM6IE9wdGlvbnNTZXJ2aWNlLFxuICAgIHByaXZhdGUgcHJldmlld1NlcnZpY2U6IFByZXZpZXdTZXJ2aWNlLFxuICAgIHByaXZhdGUgcGVybWlzc2lvbnM6IFBlcm1pc3Npb25zXG4gICkge31cblxuICBnZXQoKTogT2JzZXJ2YWJsZTxEcmF3ZXJJdGVtW10gfCBEcmF3ZXJJdGVtPiB7XG4gICAgcmV0dXJuIHRoaXMuc2hvdWxkU2hvd0JldGFQcmV2aWV3JCgpLnBpcGUoXG4gICAgICBkaXN0aW5jdFVudGlsQ2hhbmdlZCgpLFxuICAgICAgbWFwKHNob3VsZFNob3dCZXRhUHJldmlldyA9PiB7XG4gICAgICAgIGlmIChzaG91bGRTaG93QmV0YVByZXZpZXcpIHtcbiAgICAgICAgICByZXR1cm4gdGhpcy5kcmF3ZXJJdGVtO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBbXTtcbiAgICAgIH0pXG4gICAgKTtcbiAgfVxuXG4gIHByaXZhdGUgc2hvdWxkU2hvd0JldGFQcmV2aWV3JCgpOiBPYnNlcnZhYmxlPGJvb2xlYW4+IHtcbiAgICByZXR1cm4gY29tYmluZUxhdGVzdChbdGhpcy5vcHRpb25zLmdldCQoJ2hpZGVCZXRhUHJldmlldycpLCB0aGlzLnByZXZpZXdTZXJ2aWNlLml0ZW1zJF0pLnBpcGUoXG4gICAgICBtYXAoKFtoaWRlQmV0YVByZXZpZXcsIGl0ZW1zXSkgPT4ge1xuICAgICAgICBjb25zdCBoYXNJdGVtcyA9IGl0ZW1zLmxlbmd0aCA+IDA7XG4gICAgICAgIGlmICghaGFzSXRlbXMpIHtcbiAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBoYXNBZG1pblJvbGUgPSB0aGlzLnBlcm1pc3Npb25zLmhhc1JvbGUoUGVybWlzc2lvbnMuUk9MRV9URU5BTlRfTUFOQUdFTUVOVF9BRE1JTik7XG5cbiAgICAgICAgaWYgKGhpZGVCZXRhUHJldmlldyA9PT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgcmV0dXJuIGhhc0l0ZW1zICYmIGhhc0FkbWluUm9sZTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiAhaGlkZUJldGFQcmV2aWV3ICYmIGhhc0l0ZW1zICYmIGhhc0FkbWluUm9sZTtcbiAgICAgIH0pXG4gICAgKTtcbiAgfVxufVxuIl19
55
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmV0YS1wcmV2aWV3LWRyYXdlci5mYWN0b3J5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vY29yZS9iZXRhLWZlYXR1cmUvYmV0YS1wcmV2aWV3LWRyYXdlci5mYWN0b3J5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFM0MsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBRTNELE9BQU8sRUFBRSxhQUFhLEVBQWMsTUFBTSxNQUFNLENBQUM7QUFDakQsT0FBTyxFQUFFLG9CQUFvQixFQUFFLEdBQUcsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzNELE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQzdFLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUN4RCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sK0JBQStCLENBQUM7Ozs7O0FBRzVELE1BQU0sT0FBTyx3QkFBd0I7SUFRbkMsWUFDVSxPQUF1QixFQUN2QixjQUE4QixFQUM5QixXQUF3QjtRQUZ4QixZQUFPLEdBQVAsT0FBTyxDQUFnQjtRQUN2QixtQkFBYyxHQUFkLGNBQWMsQ0FBZ0I7UUFDOUIsZ0JBQVcsR0FBWCxXQUFXLENBQWE7UUFWeEIsZUFBVSxHQUFlO1lBQ2pDLFNBQVMsRUFBRSwwQkFBMEI7WUFDckMsUUFBUSxFQUFFLE9BQU87WUFDakIsUUFBUSxFQUFFLEVBQUU7WUFDWixFQUFFLEVBQUUsYUFBYTtTQUNsQixDQUFDO0lBTUMsQ0FBQztJQUVKLEdBQUc7UUFDRCxPQUFPLElBQUksQ0FBQyxzQkFBc0IsRUFBRSxDQUFDLElBQUksQ0FDdkMsb0JBQW9CLEVBQUUsRUFDdEIsR0FBRyxDQUFDLHFCQUFxQixDQUFDLEVBQUU7WUFDMUIsSUFBSSxxQkFBcUIsRUFBRSxDQUFDO2dCQUMxQixPQUFPLElBQUksQ0FBQyxVQUFVLENBQUM7WUFDekIsQ0FBQztZQUNELE9BQU8sRUFBRSxDQUFDO1FBQ1osQ0FBQyxDQUFDLENBQ0gsQ0FBQztJQUNKLENBQUM7SUFFTyxzQkFBc0I7UUFDNUIsT0FBTyxhQUFhLENBQUM7WUFDbkIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUM7WUFDcEMsSUFBSSxDQUFDLGNBQWMsQ0FBQyxxQkFBcUIsRUFBRTtTQUM1QyxDQUFDLENBQUMsSUFBSSxDQUNMLEdBQUcsQ0FBQyxDQUFDLENBQUMsZUFBZSxFQUFFLEtBQUssQ0FBQyxFQUFFLEVBQUU7WUFDL0IsTUFBTSxRQUFRLEdBQUcsS0FBSyxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUM7WUFDbEMsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO2dCQUNkLE9BQU8sS0FBSyxDQUFDO1lBQ2YsQ0FBQztZQUVELE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsT0FBTyxDQUFDLFdBQVcsQ0FBQyw0QkFBNEIsQ0FBQyxDQUFDO1lBRXhGLElBQUksZUFBZSxLQUFLLFNBQVMsRUFBRSxDQUFDO2dCQUNsQyxPQUFPLFFBQVEsSUFBSSxZQUFZLENBQUM7WUFDbEMsQ0FBQztZQUVELE9BQU8sQ0FBQyxlQUFlLElBQUksUUFBUSxJQUFJLFlBQVksQ0FBQztRQUN0RCxDQUFDLENBQUMsQ0FDSCxDQUFDO0lBQ0osQ0FBQzsrR0E5Q1Usd0JBQXdCO21IQUF4Qix3QkFBd0IsY0FEWCxNQUFNOzs0RkFDbkIsd0JBQXdCO2tCQURwQyxVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEV4dGVuc2lvbkZhY3RvcnkgfSBmcm9tICcuLi9jb21tb24vZXh0ZW5zaW9uLWhvb2tzJztcbmltcG9ydCB7IE9wdGlvbnNTZXJ2aWNlIH0gZnJvbSAnLi4vY29tbW9uL29wdGlvbnMuc2VydmljZSc7XG5pbXBvcnQgeyBEcmF3ZXJJdGVtIH0gZnJvbSAnLi4vZHJhd2VyL2RyYXdlci5tb2RlbCc7XG5pbXBvcnQgeyBjb21iaW5lTGF0ZXN0LCBPYnNlcnZhYmxlIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBkaXN0aW5jdFVudGlsQ2hhbmdlZCwgbWFwIH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHsgQmV0YVByZXZpZXdCdXR0b25Db21wb25lbnQgfSBmcm9tICcuL2JldGEtcHJldmlldy1idXR0b24uY29tcG9uZW50JztcbmltcG9ydCB7IFByZXZpZXdTZXJ2aWNlIH0gZnJvbSAnLi9iZXRhLXByZXZpZXcuc2VydmljZSc7XG5pbXBvcnQgeyBQZXJtaXNzaW9ucyB9IGZyb20gJy4uL2NvbW1vbi9wZXJtaXNzaW9ucy5zZXJ2aWNlJztcblxuQEluamVjdGFibGUoeyBwcm92aWRlZEluOiAncm9vdCcgfSlcbmV4cG9ydCBjbGFzcyBCZXRhUHJldmlld0RyYXdlckZhY3RvcnkgaW1wbGVtZW50cyBFeHRlbnNpb25GYWN0b3J5PERyYXdlckl0ZW0+IHtcbiAgcHJvdGVjdGVkIGRyYXdlckl0ZW06IERyYXdlckl0ZW0gPSB7XG4gICAgY29tcG9uZW50OiBCZXRhUHJldmlld0J1dHRvbkNvbXBvbmVudCxcbiAgICBwb3NpdGlvbjogJ3JpZ2h0JyxcbiAgICBwcmlvcml0eTogODAsXG4gICAgaWQ6ICdiZXRhUHJldmlldydcbiAgfTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIG9wdGlvbnM6IE9wdGlvbnNTZXJ2aWNlLFxuICAgIHByaXZhdGUgcHJldmlld1NlcnZpY2U6IFByZXZpZXdTZXJ2aWNlLFxuICAgIHByaXZhdGUgcGVybWlzc2lvbnM6IFBlcm1pc3Npb25zXG4gICkge31cblxuICBnZXQoKTogT2JzZXJ2YWJsZTxEcmF3ZXJJdGVtW10gfCBEcmF3ZXJJdGVtPiB7XG4gICAgcmV0dXJuIHRoaXMuc2hvdWxkU2hvd0JldGFQcmV2aWV3JCgpLnBpcGUoXG4gICAgICBkaXN0aW5jdFVudGlsQ2hhbmdlZCgpLFxuICAgICAgbWFwKHNob3VsZFNob3dCZXRhUHJldmlldyA9PiB7XG4gICAgICAgIGlmIChzaG91bGRTaG93QmV0YVByZXZpZXcpIHtcbiAgICAgICAgICByZXR1cm4gdGhpcy5kcmF3ZXJJdGVtO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBbXTtcbiAgICAgIH0pXG4gICAgKTtcbiAgfVxuXG4gIHByaXZhdGUgc2hvdWxkU2hvd0JldGFQcmV2aWV3JCgpOiBPYnNlcnZhYmxlPGJvb2xlYW4+IHtcbiAgICByZXR1cm4gY29tYmluZUxhdGVzdChbXG4gICAgICB0aGlzLm9wdGlvbnMuZ2V0JCgnaGlkZUJldGFQcmV2aWV3JyksXG4gICAgICB0aGlzLnByZXZpZXdTZXJ2aWNlLmdldEF2YWlsYWJsZUZlYXR1cmVzJCgpXG4gICAgXSkucGlwZShcbiAgICAgIG1hcCgoW2hpZGVCZXRhUHJldmlldywgaXRlbXNdKSA9PiB7XG4gICAgICAgIGNvbnN0IGhhc0l0ZW1zID0gaXRlbXMubGVuZ3RoID4gMDtcbiAgICAgICAgaWYgKCFoYXNJdGVtcykge1xuICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGhhc0FkbWluUm9sZSA9IHRoaXMucGVybWlzc2lvbnMuaGFzUm9sZShQZXJtaXNzaW9ucy5ST0xFX1RFTkFOVF9NQU5BR0VNRU5UX0FETUlOKTtcblxuICAgICAgICBpZiAoaGlkZUJldGFQcmV2aWV3ID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICByZXR1cm4gaGFzSXRlbXMgJiYgaGFzQWRtaW5Sb2xlO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuICFoaWRlQmV0YVByZXZpZXcgJiYgaGFzSXRlbXMgJiYgaGFzQWRtaW5Sb2xlO1xuICAgICAgfSlcbiAgICApO1xuICB9XG59XG4iXX0=
@@ -17,7 +17,7 @@ export class BetaPreviewComponent {
17
17
  this.betaPreviewService = inject(PreviewService);
18
18
  this.featureCacheService = inject(FeatureCacheService);
19
19
  this.markdownToHtmlPipe = inject(MarkdownToHtmlPipe);
20
- this.features$ = this.betaPreviewService.items$;
20
+ this.features$ = this.betaPreviewService.getAvailableFeatures$();
21
21
  this.shouldClose = new Promise(resolve => {
22
22
  this.resolveClose = resolve;
23
23
  });
@@ -48,17 +48,14 @@ export class BetaPreviewComponent {
48
48
  const isChecked = event.target.checked;
49
49
  await this.betaPreviewService.setFeatureActiveState(feature.key, isChecked);
50
50
  }
51
- featureKeyExists(key) {
52
- return this.featureCacheService.featureExists(key);
53
- }
54
51
  close() {
55
52
  this.resolveClose(true);
56
53
  }
57
54
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BetaPreviewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
58
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: BetaPreviewComponent, isStandalone: true, selector: "c8y-beta-preview", host: { classAttribute: "d-contents" }, providers: [MarkdownToHtmlPipe], ngImport: i0, template: "<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n <div class=\"d-flex fit-w a-i-center j-c-center\">\n <i\n class=\"icon-24 m-r-8\"\n [c8yIcon]=\"'science'\"\n ></i>\n <h3\n class=\"m-0\"\n translate\n >\n Beta features preview\n </h3>\n </div>\n <p class=\"text-16 m-t-8\">\n {{\n 'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n | translate\n }}\n </p>\n\n <p>\n {{\n 'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n | translate\n }}\n </p>\n</div>\n\n<div class=\"card-block p-0\">\n <div class=\"d-flex fit-h\">\n <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n <div\n [class.active]=\"feature === selectedFeature\"\n class=\"d-flex fit-w a-i-center\"\n *ngFor=\"let feature of features$ | async\"\n >\n <ng-container *ngIf=\"feature.key ? (featureKeyExists(feature.key) | async) : true\">\n <button\n class=\"d-flex gap-4 text-truncate flex-grow\"\n title=\"{{ feature.label }}\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n (click)=\"updateSelectedFeature(feature)\"\n >\n <i\n class=\"m-l-0\"\n [c8yIcon]=\"'science'\"\n ></i>\n {{ feature.label }}\n </button>\n <div\n class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n >\n <label\n class=\"c8y-switch c8y-switch--inline m-l-auto\"\n title=\"{{ 'Enable or disable' | translate }}\"\n >\n <input\n [attr.aria-label]=\"feature.label\"\n type=\"checkbox\"\n [checked]=\"feature.key | featureState | async\"\n (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n />\n <span class=\"slider round\"></span>\n </label>\n </div>\n </ng-container>\n </div>\n </div>\n </div>\n\n <div\n class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n *ngIf=\"selectedFeature\"\n >\n <span class=\"tag tag--primary h4\">\n {{ 'Feature preview' | translate }} \u2014 {{ selectedFeature.label }}\n </span>\n <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n type=\"button\"\n (click)=\"close()\"\n >\n {{ 'Close' | translate }}\n </button>\n</div>\n\n<ng-template #stringDescription>\n <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n", dependencies: [{ kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i2.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "pipe", type: i4.C8yTranslatePipe, name: "translate" }, { kind: "ngmodule", type: ListGroupModule }, { kind: "pipe", type: FeatureStatePipe, name: "featureState" }] }); }
55
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: BetaPreviewComponent, isStandalone: true, selector: "c8y-beta-preview", host: { classAttribute: "d-contents" }, providers: [MarkdownToHtmlPipe], ngImport: i0, template: "<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n <div class=\"d-flex fit-w a-i-center j-c-center\">\n <i\n class=\"icon-24 m-r-8\"\n [c8yIcon]=\"'science'\"\n ></i>\n <h3\n class=\"m-0\"\n translate\n >\n Beta features preview\n </h3>\n </div>\n <p class=\"text-16 m-t-8\">\n {{\n 'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n | translate\n }}\n </p>\n\n <p>\n {{\n 'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n | translate\n }}\n </p>\n</div>\n\n<div class=\"card-block p-0\">\n <div class=\"d-flex fit-h\">\n <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n <div\n [class.active]=\"feature === selectedFeature\"\n class=\"d-flex fit-w a-i-center\"\n *ngFor=\"let feature of features$ | async\"\n >\n <button\n class=\"d-flex gap-4 text-truncate flex-grow\"\n title=\"{{ feature.label }}\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n (click)=\"updateSelectedFeature(feature)\"\n >\n <i\n class=\"m-l-0\"\n [c8yIcon]=\"'science'\"\n ></i>\n {{ feature.label }}\n </button>\n <div\n class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n >\n <label\n class=\"c8y-switch c8y-switch--inline m-l-auto\"\n title=\"{{ 'Enable or disable' | translate }}\"\n >\n <input\n [attr.aria-label]=\"feature.label\"\n type=\"checkbox\"\n [checked]=\"feature.key | featureState | async\"\n (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n />\n <span class=\"slider round\"></span>\n </label>\n </div>\n </div>\n </div>\n </div>\n\n <div\n class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n *ngIf=\"selectedFeature\"\n >\n <span class=\"tag tag--primary h4\">\n {{ 'Feature preview' | translate }} \u2014 {{ selectedFeature.label }}\n </span>\n <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n type=\"button\"\n (click)=\"close()\"\n >\n {{ 'Close' | translate }}\n </button>\n</div>\n\n<ng-template #stringDescription>\n <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n", dependencies: [{ kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i2.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "pipe", type: i4.C8yTranslatePipe, name: "translate" }, { kind: "ngmodule", type: ListGroupModule }, { kind: "pipe", type: FeatureStatePipe, name: "featureState" }] }); }
59
56
  }
60
57
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BetaPreviewComponent, decorators: [{
61
58
  type: Component,
62
- args: [{ selector: 'c8y-beta-preview', host: { class: 'd-contents' }, standalone: true, imports: [NgForOf, NgIf, AsyncPipe, CommonModule, ListGroupModule, FeatureStatePipe], providers: [MarkdownToHtmlPipe], template: "<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n <div class=\"d-flex fit-w a-i-center j-c-center\">\n <i\n class=\"icon-24 m-r-8\"\n [c8yIcon]=\"'science'\"\n ></i>\n <h3\n class=\"m-0\"\n translate\n >\n Beta features preview\n </h3>\n </div>\n <p class=\"text-16 m-t-8\">\n {{\n 'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n | translate\n }}\n </p>\n\n <p>\n {{\n 'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n | translate\n }}\n </p>\n</div>\n\n<div class=\"card-block p-0\">\n <div class=\"d-flex fit-h\">\n <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n <div\n [class.active]=\"feature === selectedFeature\"\n class=\"d-flex fit-w a-i-center\"\n *ngFor=\"let feature of features$ | async\"\n >\n <ng-container *ngIf=\"feature.key ? (featureKeyExists(feature.key) | async) : true\">\n <button\n class=\"d-flex gap-4 text-truncate flex-grow\"\n title=\"{{ feature.label }}\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n (click)=\"updateSelectedFeature(feature)\"\n >\n <i\n class=\"m-l-0\"\n [c8yIcon]=\"'science'\"\n ></i>\n {{ feature.label }}\n </button>\n <div\n class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n >\n <label\n class=\"c8y-switch c8y-switch--inline m-l-auto\"\n title=\"{{ 'Enable or disable' | translate }}\"\n >\n <input\n [attr.aria-label]=\"feature.label\"\n type=\"checkbox\"\n [checked]=\"feature.key | featureState | async\"\n (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n />\n <span class=\"slider round\"></span>\n </label>\n </div>\n </ng-container>\n </div>\n </div>\n </div>\n\n <div\n class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n *ngIf=\"selectedFeature\"\n >\n <span class=\"tag tag--primary h4\">\n {{ 'Feature preview' | translate }} \u2014 {{ selectedFeature.label }}\n </span>\n <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n type=\"button\"\n (click)=\"close()\"\n >\n {{ 'Close' | translate }}\n </button>\n</div>\n\n<ng-template #stringDescription>\n <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n" }]
59
+ args: [{ selector: 'c8y-beta-preview', host: { class: 'd-contents' }, standalone: true, imports: [NgForOf, NgIf, AsyncPipe, CommonModule, ListGroupModule, FeatureStatePipe], providers: [MarkdownToHtmlPipe], template: "<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n <div class=\"d-flex fit-w a-i-center j-c-center\">\n <i\n class=\"icon-24 m-r-8\"\n [c8yIcon]=\"'science'\"\n ></i>\n <h3\n class=\"m-0\"\n translate\n >\n Beta features preview\n </h3>\n </div>\n <p class=\"text-16 m-t-8\">\n {{\n 'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n | translate\n }}\n </p>\n\n <p>\n {{\n 'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n | translate\n }}\n </p>\n</div>\n\n<div class=\"card-block p-0\">\n <div class=\"d-flex fit-h\">\n <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n <div\n [class.active]=\"feature === selectedFeature\"\n class=\"d-flex fit-w a-i-center\"\n *ngFor=\"let feature of features$ | async\"\n >\n <button\n class=\"d-flex gap-4 text-truncate flex-grow\"\n title=\"{{ feature.label }}\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n (click)=\"updateSelectedFeature(feature)\"\n >\n <i\n class=\"m-l-0\"\n [c8yIcon]=\"'science'\"\n ></i>\n {{ feature.label }}\n </button>\n <div\n class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n >\n <label\n class=\"c8y-switch c8y-switch--inline m-l-auto\"\n title=\"{{ 'Enable or disable' | translate }}\"\n >\n <input\n [attr.aria-label]=\"feature.label\"\n type=\"checkbox\"\n [checked]=\"feature.key | featureState | async\"\n (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n />\n <span class=\"slider round\"></span>\n </label>\n </div>\n </div>\n </div>\n </div>\n\n <div\n class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n *ngIf=\"selectedFeature\"\n >\n <span class=\"tag tag--primary h4\">\n {{ 'Feature preview' | translate }} \u2014 {{ selectedFeature.label }}\n </span>\n <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n type=\"button\"\n (click)=\"close()\"\n >\n {{ 'Close' | translate }}\n </button>\n</div>\n\n<ng-template #stringDescription>\n <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n" }]
63
60
  }], ctorParameters: () => [] });
64
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"beta-preview.component.js","sourceRoot":"","sources":["../../../../core/beta-feature/beta-preview.component.ts","../../../../core/beta-feature/beta-preview.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Type } from '@angular/core';\nimport { BetaFeature } from './beta-preview.model';\nimport { BehaviorSubject, firstValueFrom, Observable } from 'rxjs';\nimport { AsyncPipe, NgForOf, NgIf } from '@angular/common';\nimport { CommonModule, MarkdownToHtmlPipe } from '../common';\nimport { ListGroupModule } from '../list-group';\nimport { PreviewService } from './beta-preview.service';\nimport { FeatureStatePipe } from './feature-state.pipe';\nimport { FeatureCacheService } from './feature-cache.service';\n\n@Component({\n  selector: 'c8y-beta-preview',\n  templateUrl: './beta-preview.component.html',\n  host: { class: 'd-contents' },\n  standalone: true,\n  imports: [NgForOf, NgIf, AsyncPipe, CommonModule, ListGroupModule, FeatureStatePipe],\n  providers: [MarkdownToHtmlPipe]\n})\nexport class BetaPreviewComponent {\n  selectedFeature: BetaFeature;\n  shouldClose: Promise<boolean>;\n  description$: BehaviorSubject<string | Type<any> | Promise<string | Type<any>>> =\n    new BehaviorSubject('');\n  private resolveClose!: (value: boolean | PromiseLike<boolean>) => void;\n\n  private readonly betaPreviewService = inject(PreviewService);\n  private readonly featureCacheService = inject(FeatureCacheService);\n  private readonly markdownToHtmlPipe = inject(MarkdownToHtmlPipe);\n\n  features$: Observable<BetaFeature[]> = this.betaPreviewService.items$;\n\n  constructor() {\n    this.shouldClose = new Promise<boolean>(resolve => {\n      this.resolveClose = resolve;\n    });\n  }\n\n  async ngOnInit() {\n    await this.betaPreviewService.markAllFeaturesAsSeen();\n    const features = await firstValueFrom(this.features$);\n    this.updateSelectedFeature(features[0]);\n  }\n\n  async updateSelectedFeature(feature: BetaFeature) {\n    this.selectedFeature = feature;\n    await this.updateDescription();\n  }\n\n  async updateDescription() {\n    const description = await this.selectedFeature?.description();\n    if (!this.isComponent(description)) {\n      const html = await firstValueFrom(this.markdownToHtmlPipe.transform(description));\n      this.description$.next(html);\n    } else {\n      this.description$.next(description);\n    }\n  }\n\n  isComponent(description: string | Type<any> | Promise<string | Type<any>>) {\n    return description instanceof Function;\n  }\n\n  async onToggle(feature: BetaFeature, event: Event): Promise<void> {\n    const isChecked = (event.target as HTMLInputElement).checked;\n    await this.betaPreviewService.setFeatureActiveState(feature.key, isChecked);\n  }\n\n  featureKeyExists(key: string): Observable<boolean> {\n    return this.featureCacheService.featureExists(key);\n  }\n\n  close() {\n    this.resolveClose(true);\n  }\n}\n","<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n  <div class=\"d-flex fit-w a-i-center j-c-center\">\n    <i\n      class=\"icon-24 m-r-8\"\n      [c8yIcon]=\"'science'\"\n    ></i>\n    <h3\n      class=\"m-0\"\n      translate\n    >\n      Beta features preview\n    </h3>\n  </div>\n  <p class=\"text-16 m-t-8\">\n    {{\n      'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n        | translate\n    }}\n  </p>\n\n  <p>\n    {{\n      'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n        | translate\n    }}\n  </p>\n</div>\n\n<div class=\"card-block p-0\">\n  <div class=\"d-flex fit-h\">\n    <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n      <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n        <div\n          [class.active]=\"feature === selectedFeature\"\n          class=\"d-flex fit-w a-i-center\"\n          *ngFor=\"let feature of features$ | async\"\n        >\n          <ng-container *ngIf=\"feature.key ? (featureKeyExists(feature.key) | async) : true\">\n            <button\n              class=\"d-flex gap-4 text-truncate flex-grow\"\n              title=\"{{ feature.label }}\"\n              [ngClass]=\"{\n                'bg-component': feature === selectedFeature,\n                'bg-level-1': feature !== selectedFeature\n              }\"\n              (click)=\"updateSelectedFeature(feature)\"\n            >\n              <i\n                class=\"m-l-0\"\n                [c8yIcon]=\"'science'\"\n              ></i>\n              {{ feature.label }}\n            </button>\n            <div\n              class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n              [ngClass]=\"{\n                'bg-component': feature === selectedFeature,\n                'bg-level-1': feature !== selectedFeature\n              }\"\n            >\n              <label\n                class=\"c8y-switch c8y-switch--inline m-l-auto\"\n                title=\"{{ 'Enable or disable' | translate }}\"\n              >\n                <input\n                  [attr.aria-label]=\"feature.label\"\n                  type=\"checkbox\"\n                  [checked]=\"feature.key | featureState | async\"\n                  (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n                />\n                <span class=\"slider round\"></span>\n              </label>\n            </div>\n          </ng-container>\n        </div>\n      </div>\n    </div>\n\n    <div\n      class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n      *ngIf=\"selectedFeature\"\n    >\n      <span class=\"tag tag--primary h4\">\n        {{ 'Feature preview' | translate }} — {{ selectedFeature.label }}\n      </span>\n      <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n        <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n  <button\n    class=\"btn btn-default\"\n    title=\"{{ 'Close' | translate }}\"\n    type=\"button\"\n    (click)=\"close()\"\n  >\n    {{ 'Close' | translate }}\n  </button>\n</div>\n\n<ng-template #stringDescription>\n  <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n"]}
61
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"beta-preview.component.js","sourceRoot":"","sources":["../../../../core/beta-feature/beta-preview.component.ts","../../../../core/beta-feature/beta-preview.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Type } from '@angular/core';\nimport { BetaFeature } from './beta-preview.model';\nimport { BehaviorSubject, firstValueFrom, Observable } from 'rxjs';\nimport { AsyncPipe, NgForOf, NgIf } from '@angular/common';\nimport { CommonModule, MarkdownToHtmlPipe } from '../common';\nimport { ListGroupModule } from '../list-group';\nimport { PreviewService } from './beta-preview.service';\nimport { FeatureStatePipe } from './feature-state.pipe';\nimport { FeatureCacheService } from './feature-cache.service';\n\n@Component({\n  selector: 'c8y-beta-preview',\n  templateUrl: './beta-preview.component.html',\n  host: { class: 'd-contents' },\n  standalone: true,\n  imports: [NgForOf, NgIf, AsyncPipe, CommonModule, ListGroupModule, FeatureStatePipe],\n  providers: [MarkdownToHtmlPipe]\n})\nexport class BetaPreviewComponent {\n  selectedFeature: BetaFeature;\n  shouldClose: Promise<boolean>;\n  description$: BehaviorSubject<string | Type<any> | Promise<string | Type<any>>> =\n    new BehaviorSubject('');\n  private resolveClose!: (value: boolean | PromiseLike<boolean>) => void;\n\n  private readonly betaPreviewService = inject(PreviewService);\n  private readonly featureCacheService = inject(FeatureCacheService);\n  private readonly markdownToHtmlPipe = inject(MarkdownToHtmlPipe);\n\n  features$: Observable<BetaFeature[]> = this.betaPreviewService.getAvailableFeatures$();\n\n  constructor() {\n    this.shouldClose = new Promise<boolean>(resolve => {\n      this.resolveClose = resolve;\n    });\n  }\n\n  async ngOnInit() {\n    await this.betaPreviewService.markAllFeaturesAsSeen();\n    const features = await firstValueFrom(this.features$);\n    this.updateSelectedFeature(features[0]);\n  }\n\n  async updateSelectedFeature(feature: BetaFeature) {\n    this.selectedFeature = feature;\n    await this.updateDescription();\n  }\n\n  async updateDescription() {\n    const description = await this.selectedFeature?.description();\n    if (!this.isComponent(description)) {\n      const html = await firstValueFrom(this.markdownToHtmlPipe.transform(description));\n      this.description$.next(html);\n    } else {\n      this.description$.next(description);\n    }\n  }\n\n  isComponent(description: string | Type<any> | Promise<string | Type<any>>) {\n    return description instanceof Function;\n  }\n\n  async onToggle(feature: BetaFeature, event: Event): Promise<void> {\n    const isChecked = (event.target as HTMLInputElement).checked;\n    await this.betaPreviewService.setFeatureActiveState(feature.key, isChecked);\n  }\n\n  close() {\n    this.resolveClose(true);\n  }\n}\n","<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n  <div class=\"d-flex fit-w a-i-center j-c-center\">\n    <i\n      class=\"icon-24 m-r-8\"\n      [c8yIcon]=\"'science'\"\n    ></i>\n    <h3\n      class=\"m-0\"\n      translate\n    >\n      Beta features preview\n    </h3>\n  </div>\n  <p class=\"text-16 m-t-8\">\n    {{\n      'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n        | translate\n    }}\n  </p>\n\n  <p>\n    {{\n      'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n        | translate\n    }}\n  </p>\n</div>\n\n<div class=\"card-block p-0\">\n  <div class=\"d-flex fit-h\">\n    <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n      <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n        <div\n          [class.active]=\"feature === selectedFeature\"\n          class=\"d-flex fit-w a-i-center\"\n          *ngFor=\"let feature of features$ | async\"\n        >\n          <button\n            class=\"d-flex gap-4 text-truncate flex-grow\"\n            title=\"{{ feature.label }}\"\n            [ngClass]=\"{\n              'bg-component': feature === selectedFeature,\n              'bg-level-1': feature !== selectedFeature\n            }\"\n            (click)=\"updateSelectedFeature(feature)\"\n          >\n            <i\n              class=\"m-l-0\"\n              [c8yIcon]=\"'science'\"\n            ></i>\n            {{ feature.label }}\n          </button>\n          <div\n            class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n            [ngClass]=\"{\n              'bg-component': feature === selectedFeature,\n              'bg-level-1': feature !== selectedFeature\n            }\"\n          >\n            <label\n              class=\"c8y-switch c8y-switch--inline m-l-auto\"\n              title=\"{{ 'Enable or disable' | translate }}\"\n            >\n              <input\n                [attr.aria-label]=\"feature.label\"\n                type=\"checkbox\"\n                [checked]=\"feature.key | featureState | async\"\n                (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n              />\n              <span class=\"slider round\"></span>\n            </label>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <div\n      class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n      *ngIf=\"selectedFeature\"\n    >\n      <span class=\"tag tag--primary h4\">\n        {{ 'Feature preview' | translate }} — {{ selectedFeature.label }}\n      </span>\n      <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n        <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n      </div>\n    </div>\n  </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n  <button\n    class=\"btn btn-default\"\n    title=\"{{ 'Close' | translate }}\"\n    type=\"button\"\n    (click)=\"close()\"\n  >\n    {{ 'Close' | translate }}\n  </button>\n</div>\n\n<ng-template #stringDescription>\n  <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n"]}
@@ -19,7 +19,7 @@ export class PreviewService extends ExtensionPointForPlugins {
19
19
  this.reload$ = new BehaviorSubject(false);
20
20
  this.LOCAL_STORAGE_KEY = 'c8y_beta_features_seen';
21
21
  this.seenFeatures$ = new BehaviorSubject(this.getSeenFeatures());
22
- this.hasUnseenFeatures$ = combineLatest([this.items$, this.seenFeatures$]).pipe(map(([features, seen]) => features.some(feature => !seen.includes(feature.label))), distinctUntilChanged());
22
+ this.hasUnseenFeatures$ = combineLatest([this.getAvailableFeatures$(), this.seenFeatures$]).pipe(map(([features, seen]) => features.some(feature => !seen.includes(feature.label))), distinctUntilChanged());
23
23
  this.featureService = inject(FeatureService);
24
24
  this.featureCacheService = inject(FeatureCacheService);
25
25
  this.optionsService = inject(OptionsService);
@@ -89,6 +89,18 @@ export class PreviewService extends ExtensionPointForPlugins {
89
89
  localStorage.setItem(this.LOCAL_STORAGE_KEY, JSON.stringify(featureNames));
90
90
  this.seenFeatures$.next(featureNames);
91
91
  }
92
+ /**
93
+ * Filters the features to only include those that are available by checking the feature toggle API.
94
+ */
95
+ getAvailableFeatures$() {
96
+ return this.items$.pipe(switchMap(features => {
97
+ return combineLatest(features.map(feat => feat.key
98
+ ? this.featureCacheService
99
+ .featureExists(feat.key)
100
+ .pipe(map(exists => (exists ? feat : false)))
101
+ : of(feat))).pipe(map(features => features.filter(Boolean)));
102
+ }));
103
+ }
92
104
  /**
93
105
  *
94
106
  * @returns An array of feature names that have been seen by the user. This is retrieved from local storage.
@@ -111,4 +123,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
111
123
  type: Injectable,
112
124
  args: [{ providedIn: 'root' }]
113
125
  }], ctorParameters: () => [{ type: i0.Injector }, { type: i1.PluginsResolveService }] });
114
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"beta-preview.service.js","sourceRoot":"","sources":["../../../../core/beta-feature/beta-preview.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable, InjectionToken, Injector } from '@angular/core';\nimport { BehaviorSubject, combineLatest, firstValueFrom, Observable, of } from 'rxjs';\nimport { distinctUntilChanged, map, shareReplay, switchMap } from 'rxjs/operators';\nimport {\n  GenericHookOptions,\n  GenericHookType,\n  hookGeneric,\n  getInjectedHooks,\n  ExtensionPointForPlugins,\n  fromTriggerOnce,\n  stateToFactory,\n  sortByPriority\n} from '../common/extension-hooks';\nimport { BetaFeature } from './beta-preview.model';\nimport { PluginsResolveService } from '../plugins/plugins-resolve.service';\nimport { FeatureService } from '@c8y/client';\nimport { FeatureCacheService } from './feature-cache.service';\nimport { OptionsService } from '../common/options.service';\n\nexport const HOOK_PREVIEW = new InjectionToken<BetaFeature[]>('HOOK_PREVIEW');\n\nexport function hookPreview(\n  preview: GenericHookType<BetaFeature>,\n  options?: Partial<GenericHookOptions>\n) {\n  return hookGeneric<BetaFeature>(preview, HOOK_PREVIEW, options);\n}\n\n@Injectable({ providedIn: 'root' })\nexport class PreviewService extends ExtensionPointForPlugins<BetaFeature> {\n  items$: Observable<BetaFeature[]> = this.setupItemsObservable();\n  reload$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n  private readonly LOCAL_STORAGE_KEY = 'c8y_beta_features_seen';\n  private seenFeatures$ = new BehaviorSubject<string[]>(this.getSeenFeatures());\n  hasUnseenFeatures$ = combineLatest([this.items$, this.seenFeatures$]).pipe(\n    map(([features, seen]) => features.some(feature => !seen.includes(feature.label))),\n    distinctUntilChanged()\n  );\n  private readonly featureService = inject(FeatureService);\n  private readonly featureCacheService = inject(FeatureCacheService);\n  private readonly optionsService = inject(OptionsService);\n\n  private toggledFeaturesWithReload: Set<string> = new Set(); // track features with settings.reload = true\n\n  constructor(rootInjector: Injector, pluginService: PluginsResolveService) {\n    super(rootInjector, pluginService);\n  }\n\n  get state(): Set<BetaFeature> {\n    return this.state$.value;\n  }\n\n  /**\n   * Get the state of a specific feature.\n   * @param key The unique key for the feature.\n   * @returns Observable<boolean> representing the state of the feature.\n   */\n  getState$(key: string): Observable<boolean> {\n    return this.items$.pipe(\n      switchMap(features =>\n        this.optionsService.get$('enableAllBetaFeatures').pipe(\n          switchMap(enableAllBetaFeatures => {\n            if (enableAllBetaFeatures) {\n              return of(true);\n            }\n            const feature = features.find(f => f.key === key);\n            if (!feature) {\n              throw new Error(`Feature with key \"${key}\" does not exist.`);\n            }\n            if (feature.active$) {\n              return feature.active$;\n            }\n            return this.featureCacheService.getFeatureState(key).pipe(distinctUntilChanged());\n          })\n        )\n      ),\n      distinctUntilChanged()\n    );\n  }\n\n  /**\n   * Toggle the state of a specific feature.\n   * @param key The unique key or label for the feature. Label is used to find the custom features.\n   * @returns Promise<void> after toggling the state.\n   */\n  async setFeatureActiveState(key: string, toggleValue: boolean): Promise<void> {\n    const features = await firstValueFrom(this.items$);\n    const feature = features.find(f => f.key === key) || features.find(f => f.label === key);\n\n    if (!feature) {\n      throw new Error(`Feature with key \"${key}\" does not exist.`);\n    }\n\n    if (feature.onToggle) {\n      const success = await feature.onToggle(toggleValue);\n      if (!success) {\n        throw new Error(`Failed to toggle feature with key \"${key}\".`);\n      }\n      return;\n    }\n\n    const featureToUpdate = { key: feature.key, active: toggleValue };\n    await this.featureService.updateFeature(featureToUpdate);\n    await this.featureCacheService.resetFeatureState();\n\n    if (feature.settings?.reload) {\n      if (this.toggledFeaturesWithReload.has(key)) {\n        this.toggledFeaturesWithReload.delete(key);\n      } else {\n        this.toggledFeaturesWithReload.add(key);\n      }\n    }\n\n    this.reload$.next(this.toggledFeaturesWithReload.size > 0);\n  }\n\n  /**\n   * Marks all features as seen by the user and updates the local storage.\n   * This assumes all features are provided via the `items$` observable.\n   */\n  async markAllFeaturesAsSeen(): Promise<void> {\n    const features = await firstValueFrom(this.items$);\n    const featureNames = features.map(feature => feature.label);\n\n    localStorage.setItem(this.LOCAL_STORAGE_KEY, JSON.stringify(featureNames));\n    this.seenFeatures$.next(featureNames);\n  }\n\n  /**\n   *\n   * @returns An array of feature names that have been seen by the user. This is retrieved from local storage.\n   */\n  private getSeenFeatures(): string[] {\n    const seenFeatures = localStorage.getItem(this.LOCAL_STORAGE_KEY);\n    return seenFeatures ? JSON.parse(seenFeatures) : [];\n  }\n\n  protected setupItemsObservable(): Observable<BetaFeature[]> {\n    return fromTriggerOnce<BetaFeature>(undefined, this.refresh$, [\n      getInjectedHooks<BetaFeature>(HOOK_PREVIEW, this.injectors),\n      () => this.factories,\n      stateToFactory(this.state$)\n    ]).pipe(\n      distinctUntilChanged(),\n      map(features => sortByPriority(features)),\n      shareReplay(1)\n    );\n  }\n}\n"]}
126
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"beta-preview.service.js","sourceRoot":"","sources":["../../../../core/beta-feature/beta-preview.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { inject, Injectable, InjectionToken, Injector } from '@angular/core';\nimport { BehaviorSubject, combineLatest, firstValueFrom, Observable, of } from 'rxjs';\nimport { distinctUntilChanged, map, shareReplay, switchMap } from 'rxjs/operators';\nimport {\n  GenericHookOptions,\n  GenericHookType,\n  hookGeneric,\n  getInjectedHooks,\n  ExtensionPointForPlugins,\n  fromTriggerOnce,\n  stateToFactory,\n  sortByPriority\n} from '../common/extension-hooks';\nimport { BetaFeature } from './beta-preview.model';\nimport { PluginsResolveService } from '../plugins/plugins-resolve.service';\nimport { FeatureService } from '@c8y/client';\nimport { FeatureCacheService } from './feature-cache.service';\nimport { OptionsService } from '../common/options.service';\n\nexport const HOOK_PREVIEW = new InjectionToken<BetaFeature[]>('HOOK_PREVIEW');\n\nexport function hookPreview(\n  preview: GenericHookType<BetaFeature>,\n  options?: Partial<GenericHookOptions>\n) {\n  return hookGeneric<BetaFeature>(preview, HOOK_PREVIEW, options);\n}\n\n@Injectable({ providedIn: 'root' })\nexport class PreviewService extends ExtensionPointForPlugins<BetaFeature> {\n  items$: Observable<BetaFeature[]> = this.setupItemsObservable();\n  reload$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n  private readonly LOCAL_STORAGE_KEY = 'c8y_beta_features_seen';\n  private seenFeatures$ = new BehaviorSubject<string[]>(this.getSeenFeatures());\n  hasUnseenFeatures$ = combineLatest([this.getAvailableFeatures$(), this.seenFeatures$]).pipe(\n    map(([features, seen]) => features.some(feature => !seen.includes(feature.label))),\n    distinctUntilChanged()\n  );\n  private readonly featureService = inject(FeatureService);\n  private readonly featureCacheService = inject(FeatureCacheService);\n  private readonly optionsService = inject(OptionsService);\n\n  private toggledFeaturesWithReload: Set<string> = new Set(); // track features with settings.reload = true\n\n  constructor(rootInjector: Injector, pluginService: PluginsResolveService) {\n    super(rootInjector, pluginService);\n  }\n\n  get state(): Set<BetaFeature> {\n    return this.state$.value;\n  }\n\n  /**\n   * Get the state of a specific feature.\n   * @param key The unique key for the feature.\n   * @returns Observable<boolean> representing the state of the feature.\n   */\n  getState$(key: string): Observable<boolean> {\n    return this.items$.pipe(\n      switchMap(features =>\n        this.optionsService.get$('enableAllBetaFeatures').pipe(\n          switchMap(enableAllBetaFeatures => {\n            if (enableAllBetaFeatures) {\n              return of(true);\n            }\n            const feature = features.find(f => f.key === key);\n            if (!feature) {\n              throw new Error(`Feature with key \"${key}\" does not exist.`);\n            }\n            if (feature.active$) {\n              return feature.active$;\n            }\n            return this.featureCacheService.getFeatureState(key).pipe(distinctUntilChanged());\n          })\n        )\n      ),\n      distinctUntilChanged()\n    );\n  }\n\n  /**\n   * Toggle the state of a specific feature.\n   * @param key The unique key or label for the feature. Label is used to find the custom features.\n   * @returns Promise<void> after toggling the state.\n   */\n  async setFeatureActiveState(key: string, toggleValue: boolean): Promise<void> {\n    const features = await firstValueFrom(this.items$);\n    const feature = features.find(f => f.key === key) || features.find(f => f.label === key);\n\n    if (!feature) {\n      throw new Error(`Feature with key \"${key}\" does not exist.`);\n    }\n\n    if (feature.onToggle) {\n      const success = await feature.onToggle(toggleValue);\n      if (!success) {\n        throw new Error(`Failed to toggle feature with key \"${key}\".`);\n      }\n      return;\n    }\n\n    const featureToUpdate = { key: feature.key, active: toggleValue };\n    await this.featureService.updateFeature(featureToUpdate);\n    await this.featureCacheService.resetFeatureState();\n\n    if (feature.settings?.reload) {\n      if (this.toggledFeaturesWithReload.has(key)) {\n        this.toggledFeaturesWithReload.delete(key);\n      } else {\n        this.toggledFeaturesWithReload.add(key);\n      }\n    }\n\n    this.reload$.next(this.toggledFeaturesWithReload.size > 0);\n  }\n\n  /**\n   * Marks all features as seen by the user and updates the local storage.\n   * This assumes all features are provided via the `items$` observable.\n   */\n  async markAllFeaturesAsSeen(): Promise<void> {\n    const features = await firstValueFrom(this.items$);\n    const featureNames = features.map(feature => feature.label);\n\n    localStorage.setItem(this.LOCAL_STORAGE_KEY, JSON.stringify(featureNames));\n    this.seenFeatures$.next(featureNames);\n  }\n\n  /**\n   * Filters the features to only include those that are available by checking the feature toggle API.\n   */\n  getAvailableFeatures$(): Observable<BetaFeature[]> {\n    return this.items$.pipe(\n      switchMap(features => {\n        return combineLatest(\n          features.map(feat =>\n            feat.key\n              ? this.featureCacheService\n                  .featureExists(feat.key)\n                  .pipe(map(exists => (exists ? feat : false)))\n              : of(feat)\n          )\n        ).pipe(map(features => features.filter(Boolean))) as Observable<BetaFeature[]>;\n      })\n    );\n  }\n\n  /**\n   *\n   * @returns An array of feature names that have been seen by the user. This is retrieved from local storage.\n   */\n  private getSeenFeatures(): string[] {\n    const seenFeatures = localStorage.getItem(this.LOCAL_STORAGE_KEY);\n    return seenFeatures ? JSON.parse(seenFeatures) : [];\n  }\n\n  protected setupItemsObservable(): Observable<BetaFeature[]> {\n    return fromTriggerOnce<BetaFeature>(undefined, this.refresh$, [\n      getInjectedHooks<BetaFeature>(HOOK_PREVIEW, this.injectors),\n      () => this.factories,\n      stateToFactory(this.state$)\n    ]).pipe(\n      distinctUntilChanged(),\n      map(features => sortByPriority(features)),\n      shareReplay(1)\n    );\n  }\n}\n"]}
@@ -35,8 +35,14 @@ export class FeatureCacheService {
35
35
  this.refreshTrigger.next();
36
36
  }
37
37
  async loadFeatures() {
38
- const response = await this.featureService.list({ pageSize: 1000 });
39
- return response.data || [];
38
+ try {
39
+ const response = await this.featureService.list({ pageSize: 1000 });
40
+ return response.data || [];
41
+ }
42
+ catch (e) {
43
+ console.error('Error loading features:', e);
44
+ return [];
45
+ }
40
46
  }
41
47
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FeatureCacheService, deps: [{ token: i1.FeatureService }], target: i0.ɵɵFactoryTarget.Injectable }); }
42
48
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FeatureCacheService, providedIn: 'root' }); }
@@ -45,4 +51,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
45
51
  type: Injectable,
46
52
  args: [{ providedIn: 'root' }]
47
53
  }], ctorParameters: () => [{ type: i1.FeatureService }] });
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmVhdHVyZS1jYWNoZS5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vY29yZS9iZXRhLWZlYXR1cmUvZmVhdHVyZS1jYWNoZS5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0MsT0FBTyxFQUFFLGNBQWMsRUFBa0IsTUFBTSxhQUFhLENBQUM7QUFDN0QsT0FBTyxFQUFFLGVBQWUsRUFBRSxjQUFjLEVBQWMsTUFBTSxNQUFNLENBQUM7QUFDbkUsT0FBTyxFQUFFLEdBQUcsRUFBRSxXQUFXLEVBQUUsU0FBUyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7OztBQUU3RCxNQUFNLE9BQU8sbUJBQW1CO0lBTTlCLFlBQW9CLGNBQThCO1FBQTlCLG1CQUFjLEdBQWQsY0FBYyxDQUFnQjtRQUwxQyxtQkFBYyxHQUFHLElBQUksZUFBZSxDQUFPLFNBQVMsQ0FBQyxDQUFDO1FBQ3RELGNBQVMsR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLElBQUksQ0FDMUMsU0FBUyxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQyxFQUNwQyxXQUFXLENBQUMsQ0FBQyxDQUFDLENBQ2YsQ0FBQztJQUNtRCxDQUFDO0lBRXREOzs7T0FHRztJQUNILHNCQUFzQixDQUFDLEdBQVc7UUFDaEMsT0FBTyxjQUFjLENBQUMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsYUFBYSxDQUFDLEdBQVc7UUFDdkIsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsS0FBSyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDN0YsQ0FBQztJQUVELGVBQWUsQ0FBQyxHQUFXO1FBQ3pCLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQ3hCLEdBQUcsQ0FBQyxRQUFRLENBQUMsRUFBRTtZQUNiLE1BQU0sT0FBTyxHQUFHLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsR0FBRyxLQUFLLEdBQUcsQ0FBQyxDQUFDO1lBQ2xELE9BQU8sT0FBTyxFQUFFLE1BQU0sSUFBSSxLQUFLLENBQUM7UUFDbEMsQ0FBQyxDQUFDLENBQ0gsQ0FBQztJQUNKLENBQUM7SUFFRCxpQkFBaUI7UUFDZixJQUFJLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQzdCLENBQUM7SUFFTyxLQUFLLENBQUMsWUFBWTtRQUN4QixNQUFNLFFBQVEsR0FBRyxNQUFNLElBQUksQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7UUFDcEUsT0FBTyxRQUFRLENBQUMsSUFBSSxJQUFJLEVBQUUsQ0FBQztJQUM3QixDQUFDOytHQXpDVSxtQkFBbUI7bUhBQW5CLG1CQUFtQixjQUROLE1BQU07OzRGQUNuQixtQkFBbUI7a0JBRC9CLFVBQVU7bUJBQUMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgRmVhdHVyZVNlcnZpY2UsIElGZWF0dXJlVG9nZ2xlIH0gZnJvbSAnQGM4eS9jbGllbnQnO1xuaW1wb3J0IHsgQmVoYXZpb3JTdWJqZWN0LCBmaXJzdFZhbHVlRnJvbSwgT2JzZXJ2YWJsZSB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgbWFwLCBzaGFyZVJlcGxheSwgc3dpdGNoTWFwIH0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuQEluamVjdGFibGUoeyBwcm92aWRlZEluOiAncm9vdCcgfSlcbmV4cG9ydCBjbGFzcyBGZWF0dXJlQ2FjaGVTZXJ2aWNlIHtcbiAgcHJpdmF0ZSByZWZyZXNoVHJpZ2dlciA9IG5ldyBCZWhhdmlvclN1YmplY3Q8dm9pZD4odW5kZWZpbmVkKTtcbiAgcHJpdmF0ZSBmZWF0dXJlcyQgPSB0aGlzLnJlZnJlc2hUcmlnZ2VyLnBpcGUoXG4gICAgc3dpdGNoTWFwKCgpID0+IHRoaXMubG9hZEZlYXR1cmVzKCkpLFxuICAgIHNoYXJlUmVwbGF5KDEpXG4gICk7XG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgZmVhdHVyZVNlcnZpY2U6IEZlYXR1cmVTZXJ2aWNlKSB7fVxuXG4gIC8qKlxuICAgKiBOZWVkZWQgZm9yIHRoZSBhbmd1bGFySlMgaW1wbG1lbnRhdGlvbiB0byBzaG93L2hpZGUgc29tZSBvbGQgZmVhdHVyZXMuXG4gICAqIFJldHVybnMgYSBwcm9taXNlXG4gICAqL1xuICBnZXRGZWF0dXJlU3RhdGVQcm9taXNlKGtleTogc3RyaW5nKTogUHJvbWlzZTxib29sZWFuPiB7XG4gICAgcmV0dXJuIGZpcnN0VmFsdWVGcm9tKHRoaXMuZ2V0RmVhdHVyZVN0YXRlKGtleSkpO1xuICB9XG5cbiAgLyoqXG4gICAqXG4gICAqIEBwYXJhbSBrZXkgLSBUaGUgZmVhdHVyZSBrZXkgdG8gY2hlY2tcbiAgICogQHJldHVybnMgdHJ1ZSBpZiB0aGUgZmVhdHVyZSBrZXkgZXhpc3RzIGluIHRoZSBsaXN0IG9mIGZlYXR1cmVzLlxuICAgKi9cbiAgZmVhdHVyZUV4aXN0cyhrZXk6IHN0cmluZyk6IE9ic2VydmFibGU8Ym9vbGVhbj4ge1xuICAgIHJldHVybiB0aGlzLmZlYXR1cmVzJC5waXBlKG1hcChmZWF0dXJlcyA9PiBmZWF0dXJlcy5zb21lKGZlYXR1cmUgPT4gZmVhdHVyZS5rZXkgPT09IGtleSkpKTtcbiAgfVxuXG4gIGdldEZlYXR1cmVTdGF0ZShrZXk6IHN0cmluZyk6IE9ic2VydmFibGU8Ym9vbGVhbj4ge1xuICAgIHJldHVybiB0aGlzLmZlYXR1cmVzJC5waXBlKFxuICAgICAgbWFwKGZlYXR1cmVzID0+IHtcbiAgICAgICAgY29uc3QgZmVhdHVyZSA9IGZlYXR1cmVzLmZpbmQoZiA9PiBmLmtleSA9PT0ga2V5KTtcbiAgICAgICAgcmV0dXJuIGZlYXR1cmU/LmFjdGl2ZSB8fCBmYWxzZTtcbiAgICAgIH0pXG4gICAgKTtcbiAgfVxuXG4gIHJlc2V0RmVhdHVyZVN0YXRlKCk6IHZvaWQge1xuICAgIHRoaXMucmVmcmVzaFRyaWdnZXIubmV4dCgpO1xuICB9XG5cbiAgcHJpdmF0ZSBhc3luYyBsb2FkRmVhdHVyZXMoKTogUHJvbWlzZTxJRmVhdHVyZVRvZ2dsZVtdPiB7XG4gICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLmZlYXR1cmVTZXJ2aWNlLmxpc3QoeyBwYWdlU2l6ZTogMTAwMCB9KTtcbiAgICByZXR1cm4gcmVzcG9uc2UuZGF0YSB8fCBbXTtcbiAgfVxufVxuIl19
54
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmVhdHVyZS1jYWNoZS5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vY29yZS9iZXRhLWZlYXR1cmUvZmVhdHVyZS1jYWNoZS5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0MsT0FBTyxFQUFFLGNBQWMsRUFBa0IsTUFBTSxhQUFhLENBQUM7QUFDN0QsT0FBTyxFQUFFLGVBQWUsRUFBRSxjQUFjLEVBQWMsTUFBTSxNQUFNLENBQUM7QUFDbkUsT0FBTyxFQUFFLEdBQUcsRUFBRSxXQUFXLEVBQUUsU0FBUyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7OztBQUU3RCxNQUFNLE9BQU8sbUJBQW1CO0lBTTlCLFlBQW9CLGNBQThCO1FBQTlCLG1CQUFjLEdBQWQsY0FBYyxDQUFnQjtRQUwxQyxtQkFBYyxHQUFHLElBQUksZUFBZSxDQUFPLFNBQVMsQ0FBQyxDQUFDO1FBQ3RELGNBQVMsR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLElBQUksQ0FDMUMsU0FBUyxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQyxFQUNwQyxXQUFXLENBQUMsQ0FBQyxDQUFDLENBQ2YsQ0FBQztJQUNtRCxDQUFDO0lBRXREOzs7T0FHRztJQUNILHNCQUFzQixDQUFDLEdBQVc7UUFDaEMsT0FBTyxjQUFjLENBQUMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsYUFBYSxDQUFDLEdBQVc7UUFDdkIsT0FBTyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsS0FBSyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDN0YsQ0FBQztJQUVELGVBQWUsQ0FBQyxHQUFXO1FBQ3pCLE9BQU8sSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQ3hCLEdBQUcsQ0FBQyxRQUFRLENBQUMsRUFBRTtZQUNiLE1BQU0sT0FBTyxHQUFHLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsR0FBRyxLQUFLLEdBQUcsQ0FBQyxDQUFDO1lBQ2xELE9BQU8sT0FBTyxFQUFFLE1BQU0sSUFBSSxLQUFLLENBQUM7UUFDbEMsQ0FBQyxDQUFDLENBQ0gsQ0FBQztJQUNKLENBQUM7SUFFRCxpQkFBaUI7UUFDZixJQUFJLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQzdCLENBQUM7SUFFTyxLQUFLLENBQUMsWUFBWTtRQUN4QixJQUFJLENBQUM7WUFDSCxNQUFNLFFBQVEsR0FBRyxNQUFNLElBQUksQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7WUFDcEUsT0FBTyxRQUFRLENBQUMsSUFBSSxJQUFJLEVBQUUsQ0FBQztRQUM3QixDQUFDO1FBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNYLE9BQU8sQ0FBQyxLQUFLLENBQUMseUJBQXlCLEVBQUUsQ0FBQyxDQUFDLENBQUM7WUFDNUMsT0FBTyxFQUFFLENBQUM7UUFDWixDQUFDO0lBQ0gsQ0FBQzsrR0E5Q1UsbUJBQW1CO21IQUFuQixtQkFBbUIsY0FETixNQUFNOzs0RkFDbkIsbUJBQW1CO2tCQUQvQixVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEZlYXR1cmVTZXJ2aWNlLCBJRmVhdHVyZVRvZ2dsZSB9IGZyb20gJ0BjOHkvY2xpZW50JztcbmltcG9ydCB7IEJlaGF2aW9yU3ViamVjdCwgZmlyc3RWYWx1ZUZyb20sIE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IG1hcCwgc2hhcmVSZXBsYXksIHN3aXRjaE1hcCB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogJ3Jvb3QnIH0pXG5leHBvcnQgY2xhc3MgRmVhdHVyZUNhY2hlU2VydmljZSB7XG4gIHByaXZhdGUgcmVmcmVzaFRyaWdnZXIgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PHZvaWQ+KHVuZGVmaW5lZCk7XG4gIHByaXZhdGUgZmVhdHVyZXMkID0gdGhpcy5yZWZyZXNoVHJpZ2dlci5waXBlKFxuICAgIHN3aXRjaE1hcCgoKSA9PiB0aGlzLmxvYWRGZWF0dXJlcygpKSxcbiAgICBzaGFyZVJlcGxheSgxKVxuICApO1xuICBjb25zdHJ1Y3Rvcihwcml2YXRlIGZlYXR1cmVTZXJ2aWNlOiBGZWF0dXJlU2VydmljZSkge31cblxuICAvKipcbiAgICogTmVlZGVkIGZvciB0aGUgYW5ndWxhckpTIGltcGxtZW50YXRpb24gdG8gc2hvdy9oaWRlIHNvbWUgb2xkIGZlYXR1cmVzLlxuICAgKiBSZXR1cm5zIGEgcHJvbWlzZVxuICAgKi9cbiAgZ2V0RmVhdHVyZVN0YXRlUHJvbWlzZShrZXk6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIHJldHVybiBmaXJzdFZhbHVlRnJvbSh0aGlzLmdldEZlYXR1cmVTdGF0ZShrZXkpKTtcbiAgfVxuXG4gIC8qKlxuICAgKlxuICAgKiBAcGFyYW0ga2V5IC0gVGhlIGZlYXR1cmUga2V5IHRvIGNoZWNrXG4gICAqIEByZXR1cm5zIHRydWUgaWYgdGhlIGZlYXR1cmUga2V5IGV4aXN0cyBpbiB0aGUgbGlzdCBvZiBmZWF0dXJlcy5cbiAgICovXG4gIGZlYXR1cmVFeGlzdHMoa2V5OiBzdHJpbmcpOiBPYnNlcnZhYmxlPGJvb2xlYW4+IHtcbiAgICByZXR1cm4gdGhpcy5mZWF0dXJlcyQucGlwZShtYXAoZmVhdHVyZXMgPT4gZmVhdHVyZXMuc29tZShmZWF0dXJlID0+IGZlYXR1cmUua2V5ID09PSBrZXkpKSk7XG4gIH1cblxuICBnZXRGZWF0dXJlU3RhdGUoa2V5OiBzdHJpbmcpOiBPYnNlcnZhYmxlPGJvb2xlYW4+IHtcbiAgICByZXR1cm4gdGhpcy5mZWF0dXJlcyQucGlwZShcbiAgICAgIG1hcChmZWF0dXJlcyA9PiB7XG4gICAgICAgIGNvbnN0IGZlYXR1cmUgPSBmZWF0dXJlcy5maW5kKGYgPT4gZi5rZXkgPT09IGtleSk7XG4gICAgICAgIHJldHVybiBmZWF0dXJlPy5hY3RpdmUgfHwgZmFsc2U7XG4gICAgICB9KVxuICAgICk7XG4gIH1cblxuICByZXNldEZlYXR1cmVTdGF0ZSgpOiB2b2lkIHtcbiAgICB0aGlzLnJlZnJlc2hUcmlnZ2VyLm5leHQoKTtcbiAgfVxuXG4gIHByaXZhdGUgYXN5bmMgbG9hZEZlYXR1cmVzKCk6IFByb21pc2U8SUZlYXR1cmVUb2dnbGVbXT4ge1xuICAgIHRyeSB7XG4gICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IHRoaXMuZmVhdHVyZVNlcnZpY2UubGlzdCh7IHBhZ2VTaXplOiAxMDAwIH0pO1xuICAgICAgcmV0dXJuIHJlc3BvbnNlLmRhdGEgfHwgW107XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgY29uc29sZS5lcnJvcignRXJyb3IgbG9hZGluZyBmZWF0dXJlczonLCBlKTtcbiAgICAgIHJldHVybiBbXTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
@@ -35197,8 +35197,14 @@ class FeatureCacheService {
35197
35197
  this.refreshTrigger.next();
35198
35198
  }
35199
35199
  async loadFeatures() {
35200
- const response = await this.featureService.list({ pageSize: 1000 });
35201
- return response.data || [];
35200
+ try {
35201
+ const response = await this.featureService.list({ pageSize: 1000 });
35202
+ return response.data || [];
35203
+ }
35204
+ catch (e) {
35205
+ console.error('Error loading features:', e);
35206
+ return [];
35207
+ }
35202
35208
  }
35203
35209
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FeatureCacheService, deps: [{ token: i1.FeatureService }], target: i0.ɵɵFactoryTarget.Injectable }); }
35204
35210
  static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: FeatureCacheService, providedIn: 'root' }); }
@@ -35219,7 +35225,7 @@ class PreviewService extends ExtensionPointForPlugins {
35219
35225
  this.reload$ = new BehaviorSubject(false);
35220
35226
  this.LOCAL_STORAGE_KEY = 'c8y_beta_features_seen';
35221
35227
  this.seenFeatures$ = new BehaviorSubject(this.getSeenFeatures());
35222
- this.hasUnseenFeatures$ = combineLatest([this.items$, this.seenFeatures$]).pipe(map(([features, seen]) => features.some(feature => !seen.includes(feature.label))), distinctUntilChanged());
35228
+ this.hasUnseenFeatures$ = combineLatest([this.getAvailableFeatures$(), this.seenFeatures$]).pipe(map(([features, seen]) => features.some(feature => !seen.includes(feature.label))), distinctUntilChanged());
35223
35229
  this.featureService = inject(FeatureService);
35224
35230
  this.featureCacheService = inject(FeatureCacheService);
35225
35231
  this.optionsService = inject(OptionsService);
@@ -35289,6 +35295,18 @@ class PreviewService extends ExtensionPointForPlugins {
35289
35295
  localStorage.setItem(this.LOCAL_STORAGE_KEY, JSON.stringify(featureNames));
35290
35296
  this.seenFeatures$.next(featureNames);
35291
35297
  }
35298
+ /**
35299
+ * Filters the features to only include those that are available by checking the feature toggle API.
35300
+ */
35301
+ getAvailableFeatures$() {
35302
+ return this.items$.pipe(switchMap(features => {
35303
+ return combineLatest(features.map(feat => feat.key
35304
+ ? this.featureCacheService
35305
+ .featureExists(feat.key)
35306
+ .pipe(map(exists => (exists ? feat : false)))
35307
+ : of(feat))).pipe(map(features => features.filter(Boolean)));
35308
+ }));
35309
+ }
35292
35310
  /**
35293
35311
  *
35294
35312
  * @returns An array of feature names that have been seen by the user. This is retrieved from local storage.
@@ -35338,7 +35356,7 @@ class BetaPreviewComponent {
35338
35356
  this.betaPreviewService = inject(PreviewService);
35339
35357
  this.featureCacheService = inject(FeatureCacheService);
35340
35358
  this.markdownToHtmlPipe = inject(MarkdownToHtmlPipe);
35341
- this.features$ = this.betaPreviewService.items$;
35359
+ this.features$ = this.betaPreviewService.getAvailableFeatures$();
35342
35360
  this.shouldClose = new Promise(resolve => {
35343
35361
  this.resolveClose = resolve;
35344
35362
  });
@@ -35369,18 +35387,15 @@ class BetaPreviewComponent {
35369
35387
  const isChecked = event.target.checked;
35370
35388
  await this.betaPreviewService.setFeatureActiveState(feature.key, isChecked);
35371
35389
  }
35372
- featureKeyExists(key) {
35373
- return this.featureCacheService.featureExists(key);
35374
- }
35375
35390
  close() {
35376
35391
  this.resolveClose(true);
35377
35392
  }
35378
35393
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BetaPreviewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
35379
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: BetaPreviewComponent, isStandalone: true, selector: "c8y-beta-preview", host: { classAttribute: "d-contents" }, providers: [MarkdownToHtmlPipe], ngImport: i0, template: "<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n <div class=\"d-flex fit-w a-i-center j-c-center\">\n <i\n class=\"icon-24 m-r-8\"\n [c8yIcon]=\"'science'\"\n ></i>\n <h3\n class=\"m-0\"\n translate\n >\n Beta features preview\n </h3>\n </div>\n <p class=\"text-16 m-t-8\">\n {{\n 'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n | translate\n }}\n </p>\n\n <p>\n {{\n 'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n | translate\n }}\n </p>\n</div>\n\n<div class=\"card-block p-0\">\n <div class=\"d-flex fit-h\">\n <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n <div\n [class.active]=\"feature === selectedFeature\"\n class=\"d-flex fit-w a-i-center\"\n *ngFor=\"let feature of features$ | async\"\n >\n <ng-container *ngIf=\"feature.key ? (featureKeyExists(feature.key) | async) : true\">\n <button\n class=\"d-flex gap-4 text-truncate flex-grow\"\n title=\"{{ feature.label }}\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n (click)=\"updateSelectedFeature(feature)\"\n >\n <i\n class=\"m-l-0\"\n [c8yIcon]=\"'science'\"\n ></i>\n {{ feature.label }}\n </button>\n <div\n class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n >\n <label\n class=\"c8y-switch c8y-switch--inline m-l-auto\"\n title=\"{{ 'Enable or disable' | translate }}\"\n >\n <input\n [attr.aria-label]=\"feature.label\"\n type=\"checkbox\"\n [checked]=\"feature.key | featureState | async\"\n (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n />\n <span class=\"slider round\"></span>\n </label>\n </div>\n </ng-container>\n </div>\n </div>\n </div>\n\n <div\n class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n *ngIf=\"selectedFeature\"\n >\n <span class=\"tag tag--primary h4\">\n {{ 'Feature preview' | translate }} \u2014 {{ selectedFeature.label }}\n </span>\n <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n type=\"button\"\n (click)=\"close()\"\n >\n {{ 'Close' | translate }}\n </button>\n</div>\n\n<ng-template #stringDescription>\n <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n", dependencies: [{ kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i2$1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2$1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }, { kind: "ngmodule", type: ListGroupModule }, { kind: "pipe", type: FeatureStatePipe, name: "featureState" }] }); }
35394
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: BetaPreviewComponent, isStandalone: true, selector: "c8y-beta-preview", host: { classAttribute: "d-contents" }, providers: [MarkdownToHtmlPipe], ngImport: i0, template: "<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n <div class=\"d-flex fit-w a-i-center j-c-center\">\n <i\n class=\"icon-24 m-r-8\"\n [c8yIcon]=\"'science'\"\n ></i>\n <h3\n class=\"m-0\"\n translate\n >\n Beta features preview\n </h3>\n </div>\n <p class=\"text-16 m-t-8\">\n {{\n 'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n | translate\n }}\n </p>\n\n <p>\n {{\n 'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n | translate\n }}\n </p>\n</div>\n\n<div class=\"card-block p-0\">\n <div class=\"d-flex fit-h\">\n <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n <div\n [class.active]=\"feature === selectedFeature\"\n class=\"d-flex fit-w a-i-center\"\n *ngFor=\"let feature of features$ | async\"\n >\n <button\n class=\"d-flex gap-4 text-truncate flex-grow\"\n title=\"{{ feature.label }}\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n (click)=\"updateSelectedFeature(feature)\"\n >\n <i\n class=\"m-l-0\"\n [c8yIcon]=\"'science'\"\n ></i>\n {{ feature.label }}\n </button>\n <div\n class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n >\n <label\n class=\"c8y-switch c8y-switch--inline m-l-auto\"\n title=\"{{ 'Enable or disable' | translate }}\"\n >\n <input\n [attr.aria-label]=\"feature.label\"\n type=\"checkbox\"\n [checked]=\"feature.key | featureState | async\"\n (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n />\n <span class=\"slider round\"></span>\n </label>\n </div>\n </div>\n </div>\n </div>\n\n <div\n class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n *ngIf=\"selectedFeature\"\n >\n <span class=\"tag tag--primary h4\">\n {{ 'Feature preview' | translate }} \u2014 {{ selectedFeature.label }}\n </span>\n <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n type=\"button\"\n (click)=\"close()\"\n >\n {{ 'Close' | translate }}\n </button>\n</div>\n\n<ng-template #stringDescription>\n <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n", dependencies: [{ kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: AsyncPipe, name: "async" }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i2$1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2$1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "pipe", type: C8yTranslatePipe, name: "translate" }, { kind: "ngmodule", type: ListGroupModule }, { kind: "pipe", type: FeatureStatePipe, name: "featureState" }] }); }
35380
35395
  }
35381
35396
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: BetaPreviewComponent, decorators: [{
35382
35397
  type: Component,
35383
- args: [{ selector: 'c8y-beta-preview', host: { class: 'd-contents' }, standalone: true, imports: [NgForOf, NgIf, AsyncPipe, CommonModule, ListGroupModule, FeatureStatePipe], providers: [MarkdownToHtmlPipe], template: "<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n <div class=\"d-flex fit-w a-i-center j-c-center\">\n <i\n class=\"icon-24 m-r-8\"\n [c8yIcon]=\"'science'\"\n ></i>\n <h3\n class=\"m-0\"\n translate\n >\n Beta features preview\n </h3>\n </div>\n <p class=\"text-16 m-t-8\">\n {{\n 'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n | translate\n }}\n </p>\n\n <p>\n {{\n 'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n | translate\n }}\n </p>\n</div>\n\n<div class=\"card-block p-0\">\n <div class=\"d-flex fit-h\">\n <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n <div\n [class.active]=\"feature === selectedFeature\"\n class=\"d-flex fit-w a-i-center\"\n *ngFor=\"let feature of features$ | async\"\n >\n <ng-container *ngIf=\"feature.key ? (featureKeyExists(feature.key) | async) : true\">\n <button\n class=\"d-flex gap-4 text-truncate flex-grow\"\n title=\"{{ feature.label }}\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n (click)=\"updateSelectedFeature(feature)\"\n >\n <i\n class=\"m-l-0\"\n [c8yIcon]=\"'science'\"\n ></i>\n {{ feature.label }}\n </button>\n <div\n class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n >\n <label\n class=\"c8y-switch c8y-switch--inline m-l-auto\"\n title=\"{{ 'Enable or disable' | translate }}\"\n >\n <input\n [attr.aria-label]=\"feature.label\"\n type=\"checkbox\"\n [checked]=\"feature.key | featureState | async\"\n (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n />\n <span class=\"slider round\"></span>\n </label>\n </div>\n </ng-container>\n </div>\n </div>\n </div>\n\n <div\n class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n *ngIf=\"selectedFeature\"\n >\n <span class=\"tag tag--primary h4\">\n {{ 'Feature preview' | translate }} \u2014 {{ selectedFeature.label }}\n </span>\n <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n type=\"button\"\n (click)=\"close()\"\n >\n {{ 'Close' | translate }}\n </button>\n</div>\n\n<ng-template #stringDescription>\n <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n" }]
35398
+ args: [{ selector: 'c8y-beta-preview', host: { class: 'd-contents' }, standalone: true, imports: [NgForOf, NgIf, AsyncPipe, CommonModule, ListGroupModule, FeatureStatePipe], providers: [MarkdownToHtmlPipe], template: "<div class=\"card-block separator-bottom bg-component text-center flex-no-shrink p-24\">\n <div class=\"d-flex fit-w a-i-center j-c-center\">\n <i\n class=\"icon-24 m-r-8\"\n [c8yIcon]=\"'science'\"\n ></i>\n <h3\n class=\"m-0\"\n translate\n >\n Beta features preview\n </h3>\n </div>\n <p class=\"text-16 m-t-8\">\n {{\n 'Select any feature to view additional details and documentation. Toggle features on or off using the switches.'\n | translate\n }}\n </p>\n\n <p>\n {{\n 'These beta capabilities may change or be removed before final release. We value your feedback. The descriptions for the features below are available only in English.'\n | translate\n }}\n </p>\n</div>\n\n<div class=\"card-block p-0\">\n <div class=\"d-flex fit-h\">\n <div class=\"col-md-4 col-lg-3 p-l-0 p-r-0 sticky-top m-t-0 inner-scroll bg-level-1\">\n <div class=\"nav-tabs nav-tabs-vertical nav-tabs-vertical--wide p-t-0\">\n <div\n [class.active]=\"feature === selectedFeature\"\n class=\"d-flex fit-w a-i-center\"\n *ngFor=\"let feature of features$ | async\"\n >\n <button\n class=\"d-flex gap-4 text-truncate flex-grow\"\n title=\"{{ feature.label }}\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n (click)=\"updateSelectedFeature(feature)\"\n >\n <i\n class=\"m-l-0\"\n [c8yIcon]=\"'science'\"\n ></i>\n {{ feature.label }}\n </button>\n <div\n class=\"a-s-stretch d-flex a-i-center separator-bottom p-r-16\"\n [ngClass]=\"{\n 'bg-component': feature === selectedFeature,\n 'bg-level-1': feature !== selectedFeature\n }\"\n >\n <label\n class=\"c8y-switch c8y-switch--inline m-l-auto\"\n title=\"{{ 'Enable or disable' | translate }}\"\n >\n <input\n [attr.aria-label]=\"feature.label\"\n type=\"checkbox\"\n [checked]=\"feature.key | featureState | async\"\n (change)=\"$event.stopPropagation(); onToggle(feature, $event)\"\n />\n <span class=\"slider round\"></span>\n </label>\n </div>\n </div>\n </div>\n </div>\n\n <div\n class=\"col-md-8 col-lg-9 p-24 inner-scroll\"\n *ngIf=\"selectedFeature\"\n >\n <span class=\"tag tag--primary h4\">\n {{ 'Feature preview' | translate }} \u2014 {{ selectedFeature.label }}\n </span>\n <div *ngIf=\"isComponent(description$.value); else stringDescription\">\n <ng-container *ngComponentOutlet=\"description$.value\"></ng-container>\n </div>\n </div>\n </div>\n</div>\n\n<div class=\"text-center card-footer p-24 separator\">\n <button\n class=\"btn btn-default\"\n title=\"{{ 'Close' | translate }}\"\n type=\"button\"\n (click)=\"close()\"\n >\n {{ 'Close' | translate }}\n </button>\n</div>\n\n<ng-template #stringDescription>\n <p [innerHTML]=\"description$.value\"></p>\n</ng-template>\n" }]
35384
35399
  }], ctorParameters: () => [] });
35385
35400
 
35386
35401
  class BetaPreviewButtonComponent {
@@ -35448,7 +35463,10 @@ class BetaPreviewDrawerFactory {
35448
35463
  }));
35449
35464
  }
35450
35465
  shouldShowBetaPreview$() {
35451
- return combineLatest([this.options.get$('hideBetaPreview'), this.previewService.items$]).pipe(map(([hideBetaPreview, items]) => {
35466
+ return combineLatest([
35467
+ this.options.get$('hideBetaPreview'),
35468
+ this.previewService.getAvailableFeatures$()
35469
+ ]).pipe(map(([hideBetaPreview, items]) => {
35452
35470
  const hasItems = items.length > 0;
35453
35471
  if (!hasItems) {
35454
35472
  return false;