@c8y/ngx-components 1021.22.160 → 1021.22.162

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.
Files changed (32) hide show
  1. package/core/dynamic-forms/textarea/textarea.type.component.d.ts +1 -0
  2. package/core/dynamic-forms/textarea/textarea.type.component.d.ts.map +1 -1
  3. package/esm2022/core/dynamic-forms/textarea/textarea.type.component.mjs +8 -7
  4. package/esm2022/datapoints-export-selector/datapoints-export-selector-modal/datapoints-export-selector-file-exporter/data-fetching.service.mjs +7 -7
  5. package/esm2022/datapoints-export-selector/datapoints-export-selector-modal/datapoints-export-selector-file-exporter/datapoints-export-selector-preview/datapoints-export-selector-preview.component.mjs +3 -3
  6. package/esm2022/device-grid/columns/type.device-grid-column.mjs +2 -2
  7. package/esm2022/device-list/add-smart-group.component.mjs +3 -3
  8. package/esm2022/static-assets/data/static-assets.service.mjs +11 -8
  9. package/fesm2022/c8y-ngx-components-datapoints-export-selector.mjs +8 -8
  10. package/fesm2022/c8y-ngx-components-datapoints-export-selector.mjs.map +1 -1
  11. package/fesm2022/c8y-ngx-components-device-grid.mjs +1 -1
  12. package/fesm2022/c8y-ngx-components-device-grid.mjs.map +1 -1
  13. package/fesm2022/c8y-ngx-components-device-list.mjs +2 -2
  14. package/fesm2022/c8y-ngx-components-device-list.mjs.map +1 -1
  15. package/fesm2022/c8y-ngx-components-static-assets-data.mjs +9 -5
  16. package/fesm2022/c8y-ngx-components-static-assets-data.mjs.map +1 -1
  17. package/fesm2022/c8y-ngx-components.mjs +5 -5
  18. package/fesm2022/c8y-ngx-components.mjs.map +1 -1
  19. package/locales/de.po +47 -108
  20. package/locales/es.po +30 -43
  21. package/locales/fr.po +50 -96
  22. package/locales/ja_JP.po +24 -48
  23. package/locales/ko.po +32 -46
  24. package/locales/locales.pot +23 -29
  25. package/locales/nl.po +32 -43
  26. package/locales/pl.po +31 -44
  27. package/locales/pt_BR.po +33 -45
  28. package/locales/zh_CN.po +33 -46
  29. package/locales/zh_TW.po +33 -46
  30. package/package.json +1 -1
  31. package/static-assets/data/static-assets.service.d.ts +1 -2
  32. package/static-assets/data/static-assets.service.d.ts.map +1 -1
@@ -1,6 +1,5 @@
1
1
  import { Injectable } from '@angular/core';
2
- import { ApplicationAvailability, ApplicationType, FetchClient } from '@c8y/client';
3
- import { ApplicationService } from '@c8y/client';
2
+ import { ApplicationAvailability, ApplicationService, ApplicationType, FetchClient } from '@c8y/client';
4
3
  import { AppStateService, FilesService, ZipService } from '@c8y/ngx-components';
5
4
  import * as i0 from "@angular/core";
6
5
  import * as i1 from "@c8y/ngx-components";
@@ -141,12 +140,16 @@ export class StaticAssetsService {
141
140
  * Gets the static assets app for the given type.
142
141
  */
143
142
  async getAppForTenant(type) {
144
- const appName = this.getContextPath(type, false);
145
- const { data: apps } = await this.appService.listByName(appName);
146
- if (!apps.length) {
147
- throw Error(`No app with name ${appName} found.`);
143
+ const contextPath = this.getContextPath(type);
144
+ const { data: manifest } = await this.appService.getManifestOfContextPath(contextPath);
145
+ if (!manifest.id) {
146
+ throw Error(`No app with path ${contextPath} found.`);
148
147
  }
149
- return apps[0];
148
+ const { data: app } = await this.appService.detail(manifest.id);
149
+ if (app.owner?.tenant.id !== this.appstate.currentTenant.value?.name) {
150
+ throw Error(`No app with path ${contextPath} found that is owned by the current tenant.`);
151
+ }
152
+ return app;
150
153
  }
151
154
  getContentJSONForType(type, appendTenantId) {
152
155
  const path = this.getPathForContent(type, appendTenantId);
@@ -253,4 +256,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
253
256
  type: Injectable,
254
257
  args: [{ providedIn: 'root' }]
255
258
  }], ctorParameters: () => [{ type: i1.AppStateService }, { type: i2.ApplicationService }, { type: i2.FetchClient }, { type: i1.FilesService }, { type: i1.ZipService }] });
256
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"static-assets.service.js","sourceRoot":"","sources":["../../../../static-assets/data/static-assets.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { ApplicationAvailability, ApplicationType, FetchClient, IApplication } from '@c8y/client';\nimport { ApplicationService } from '@c8y/client';\nimport { AppStateService, DroppedFile, FilesService, ZipService } from '@c8y/ngx-components';\nimport { StaticAsset, StaticAssetType } from './static-assets.model';\n\n@Injectable({ providedIn: 'root' })\nexport class StaticAssetsService {\n  readonly fileNames = {\n    contents: 'contents.json',\n    manifest: 'cumulocity.json',\n    exportZipName: 'static-assets.zip'\n  } as const;\n  private listFilesCachePromises: Record<string, Promise<StaticAsset[]>> = {};\n\n  constructor(\n    private appstate: AppStateService,\n    private appService: ApplicationService,\n    private fetchClient: FetchClient,\n    private fileService: FilesService,\n    private zip: ZipService\n  ) {}\n\n  /**\n   * Lists all files of the given type.\n   * Forces a cache renewal.\n   */\n  async listFiles<T extends string = StaticAssetType>(type: T): Promise<StaticAsset[]> {\n    return this.listFilesCached(type, true);\n  }\n\n  /**\n   * Lists all files of the given type.\n   * Uses a cache.\n   */\n  async listFilesCached<T extends string = StaticAssetType>(\n    type: T,\n    forceCacheRenewal = false\n  ): Promise<StaticAsset[]> {\n    if (!this.listFilesCachePromises[type] || forceCacheRenewal) {\n      this.listFilesCachePromises[type] = this._listFiles(type);\n    }\n\n    return await this.listFilesCachePromises[type];\n  }\n\n  /**\n   * Clones all assets of the given type from parent tenants into the current tenant.\n   */\n  async cloneAssetsIntoTenant<T extends string = StaticAssetType>(type: T) {\n    const { data: apps } = await this.appService.list({\n      availability: ApplicationAvailability.SHARED,\n      type: ApplicationType.HOSTED,\n      pageSize: 2000\n    });\n\n    const tenantIdsFromOtherAssetApps = apps\n      .filter(app => {\n        const result =\n          app.owner?.tenant?.id &&\n          app.owner?.tenant?.id !== this.appstate.currentTenant.value?.name &&\n          app.contextPath === this.getContextPath(type, app.owner?.tenant?.id);\n        return result;\n      })\n      .map(app => app.owner?.tenant?.id);\n\n    let filesOfCurrentTenant = await this._listFiles(type);\n\n    let filesToUpload = new Array<File>();\n\n    const oldAssets = new Array<StaticAsset>();\n\n    for (const tenantId of tenantIdsFromOtherAssetApps) {\n      try {\n        const filesOfApp = await this.getContentJSONForType(type, tenantId);\n        oldAssets.push(...filesOfApp);\n        for (const file of filesOfApp) {\n          try {\n            const response = await this.fetchClient.fetch(file.path);\n            if (response.status !== 200) {\n              console.warn(`Failed to get file \"${file.fileName}\" from path ${file.path}`);\n              continue;\n            }\n            const fileContent = await response.blob();\n\n            // remove existing file with same name\n            filesOfCurrentTenant = filesOfCurrentTenant.filter(\n              existingFile => existingFile.fileName !== file.fileName\n            );\n            filesToUpload = filesToUpload.filter(\n              existingFile => existingFile.name !== file.fileName\n            );\n\n            filesToUpload.push(new File([fileContent], file.fileName, { type: file.type }));\n          } catch (e) {\n            console.warn(`Failed to add file \"${file.fileName}\" from tenant ${tenantId}`, e);\n          }\n        }\n      } catch (e) {\n        console.warn(`Failed to get asset files from tenant ${tenantId}`);\n      }\n    }\n\n    if (!filesToUpload.length) {\n      return;\n    }\n\n    const newAssets = await this.addFilesToStaticAssets(type, filesToUpload, filesOfCurrentTenant);\n\n    return { oldAssets, newAssets };\n  }\n\n  /**\n   * Adds the given files to the static assets of the given type.\n   */\n  async addFilesToStaticAssets<T extends string = StaticAssetType>(\n    type: T,\n    files: DroppedFile[] | File[],\n    existingFiles: StaticAsset[],\n    throwErrorOnExistingFiles = true\n  ) {\n    const app = await this.getAppForTenant(type);\n    const filesToAddToContents = new Array<StaticAsset>();\n    const addedAt = new Date().toISOString();\n    const filesToUpload: {\n      path: string;\n      contents: Blob;\n    }[] = [];\n    for (const droppedFile of files) {\n      const file = droppedFile instanceof File ? droppedFile : droppedFile.file;\n      const fileName = file.name.toLowerCase().replace(/[^a-zA-Z0-9\\-\\_\\.]/g, '');\n      const newFile = new File([file], fileName, {\n        type: file.type,\n        lastModified: file.lastModified\n      });\n      if (existingFiles.find(existingFile => existingFile.fileName === fileName)) {\n        if (throwErrorOnExistingFiles) {\n          throw Error(`File with name \"${fileName}\" is already existing.`);\n        }\n        existingFiles = existingFiles.filter(existingFile => existingFile.fileName !== fileName);\n      }\n      const fileExtension = file.name.split('.').pop();\n      filesToAddToContents.push({\n        addedAt,\n        lastModified: file.lastModified,\n        extension: fileExtension,\n        fileName: fileName,\n        originalFileName: file.name,\n        path: `/apps/public/${app.contextPath}/${fileName}`,\n        size: newFile.size,\n        type: newFile.type,\n        hashSum: await this.fileService.getHashSumOfFile(newFile)\n      });\n      filesToUpload.push({\n        contents: newFile,\n        path: fileName\n      });\n    }\n\n    const newFiles: StaticAsset[] = [...existingFiles, ...filesToAddToContents];\n    filesToUpload.push({\n      contents: new File([JSON.stringify(newFiles)], this.fileNames.contents, {\n        type: 'application/json'\n      }),\n      path: this.fileNames.contents\n    });\n    await this.appService.binary(app).updateFiles(filesToUpload);\n    await this.ensureAddedFilesArePresent(filesToAddToContents);\n    return newFiles;\n  }\n\n  /**\n   * Gets the static assets app for the given type.\n   */\n  async getAppForTenant<T extends string = StaticAssetType>(type: T): Promise<IApplication> {\n    const appName = this.getContextPath(type, false);\n    const { data: apps } = await this.appService.listByName(appName);\n    if (!apps.length) {\n      throw Error(`No app with name ${appName} found.`);\n    }\n    return apps[0];\n  }\n\n  private getContentJSONForType<T extends string = StaticAssetType>(\n    type: T,\n    appendTenantId?: string | null | undefined\n  ) {\n    const path = this.getPathForContent(type, appendTenantId);\n    return this.getContentJSONFromPath(path);\n  }\n\n  private async getContentJSONFromPath(path: string): Promise<StaticAsset[]> {\n    const response = await this.fetchClient.fetch(path);\n    if (response.status !== 200) {\n      throw Error(`Failed to retrieve ${path}`);\n    }\n\n    const content = await response.json();\n    return content;\n  }\n\n  private async ensureAddedFilesArePresent(files: StaticAsset[]) {\n    for (const file of files) {\n      await this.ensureFileIsPresent(file);\n    }\n  }\n\n  private async ensureFileIsPresent(file: StaticAsset) {\n    for (let i = 0; i < 12; i++) {\n      try {\n        const currentDate = new Date();\n        const response = await this.fetchClient.fetch(\n          `${file.path}?nocache=${currentDate.getTime()}`\n        );\n        if (response.status !== 200) {\n          await this.sleep(5_000);\n          continue;\n        }\n\n        const blob = await response.blob();\n        const hashSum = await this.fileService.getHashSumOfFile(blob);\n        if (hashSum !== file.hashSum) {\n          await this.sleep(5_000);\n          continue;\n        }\n        return;\n      } catch (e) {\n        console.warn(e);\n        continue;\n      }\n    }\n    throw Error(`Unable to retrieve file from ${file.path}`);\n  }\n\n  private sleep(duration: number): Promise<void> {\n    return new Promise<void>(resolve => setTimeout(() => resolve(), duration));\n  }\n\n  private async _listFiles<T extends string = StaticAssetType>(type: T) {\n    try {\n      const files = await this.getContentJSONForType(type);\n      return files;\n    } catch (e) {\n      console.warn(e);\n    }\n\n    return await this.createEmptyApp(type);\n  }\n\n  private getPathForContent<T extends string = StaticAssetType>(\n    type: T,\n    appendTenantId?: string | null | undefined\n  ): string {\n    const currentDate = new Date();\n    const contextPath = this.getContextPath(type, appendTenantId);\n    return `/apps/public/${contextPath}/${\n      this.fileNames.contents\n    }?nocache=${currentDate.getTime()}`;\n  }\n\n  private getContextPath<T extends string = StaticAssetType>(\n    type: T,\n    appendTenantId?: string | false | undefined\n  ) {\n    let contextPath: string = type + '-assets';\n    const tenantId =\n      appendTenantId === undefined ? this.appstate.currentTenant.value?.name : appendTenantId;\n    if (tenantId) {\n      contextPath = `${contextPath}-${tenantId}`;\n    }\n    return contextPath;\n  }\n\n  private async createEmptyApp<T extends string = StaticAssetType>(\n    type: T,\n    appendTenantId?: string | null | undefined\n  ): Promise<StaticAsset[]> {\n    const name = this.getContextPath(type, false);\n    const contextPath = this.getContextPath(type, appendTenantId);\n    const key = `${contextPath}-key`;\n    const manifest: IApplication = {\n      name,\n      contextPath,\n      key,\n      description: `Application containing static assets used for ${type}.`,\n      noAppSwitcher: true,\n      type: ApplicationType.HOSTED,\n      availability: ApplicationAvailability.MARKET\n    };\n\n    const content = [];\n    const zipFile = await this.zip.createZip([\n      {\n        fileName: `${this.fileNames.manifest}`,\n        blob: new Blob([JSON.stringify(manifest)])\n      },\n      {\n        fileName: `${this.fileNames.contents}`,\n        blob: new Blob([JSON.stringify(content)])\n      }\n    ]);\n    const { data: app } = await this.appService.create(manifest);\n    const { data: appBinary } = await this.appService\n      .binary(app)\n      .upload(zipFile, `empty-${name}.zip`);\n    await this.appService.update({ id: app.id, activeVersionId: appBinary.id as string });\n    return [];\n  }\n}\n"]}
259
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"static-assets.service.js","sourceRoot":"","sources":["../../../../static-assets/data/static-assets.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport {\n  ApplicationAvailability,\n  ApplicationService,\n  ApplicationType,\n  FetchClient,\n  IApplication\n} from '@c8y/client';\nimport { AppStateService, DroppedFile, FilesService, ZipService } from '@c8y/ngx-components';\nimport { StaticAsset, StaticAssetType } from './static-assets.model';\n\n@Injectable({ providedIn: 'root' })\nexport class StaticAssetsService {\n  readonly fileNames = {\n    contents: 'contents.json',\n    manifest: 'cumulocity.json',\n    exportZipName: 'static-assets.zip'\n  } as const;\n  private listFilesCachePromises: Record<string, Promise<StaticAsset[]>> = {};\n\n  constructor(\n    private appstate: AppStateService,\n    private appService: ApplicationService,\n    private fetchClient: FetchClient,\n    private fileService: FilesService,\n    private zip: ZipService\n  ) {}\n\n  /**\n   * Lists all files of the given type.\n   * Forces a cache renewal.\n   */\n  async listFiles<T extends string = StaticAssetType>(type: T): Promise<StaticAsset[]> {\n    return this.listFilesCached(type, true);\n  }\n\n  /**\n   * Lists all files of the given type.\n   * Uses a cache.\n   */\n  async listFilesCached<T extends string = StaticAssetType>(\n    type: T,\n    forceCacheRenewal = false\n  ): Promise<StaticAsset[]> {\n    if (!this.listFilesCachePromises[type] || forceCacheRenewal) {\n      this.listFilesCachePromises[type] = this._listFiles(type);\n    }\n\n    return await this.listFilesCachePromises[type];\n  }\n\n  /**\n   * Clones all assets of the given type from parent tenants into the current tenant.\n   */\n  async cloneAssetsIntoTenant<T extends string = StaticAssetType>(type: T) {\n    const { data: apps } = await this.appService.list({\n      availability: ApplicationAvailability.SHARED,\n      type: ApplicationType.HOSTED,\n      pageSize: 2000\n    });\n\n    const tenantIdsFromOtherAssetApps = apps\n      .filter(app => {\n        const result =\n          app.owner?.tenant?.id &&\n          app.owner?.tenant?.id !== this.appstate.currentTenant.value?.name &&\n          app.contextPath === this.getContextPath(type, app.owner?.tenant?.id);\n        return result;\n      })\n      .map(app => app.owner?.tenant?.id);\n\n    let filesOfCurrentTenant = await this._listFiles(type);\n\n    let filesToUpload = new Array<File>();\n\n    const oldAssets = new Array<StaticAsset>();\n\n    for (const tenantId of tenantIdsFromOtherAssetApps) {\n      try {\n        const filesOfApp = await this.getContentJSONForType(type, tenantId);\n        oldAssets.push(...filesOfApp);\n        for (const file of filesOfApp) {\n          try {\n            const response = await this.fetchClient.fetch(file.path);\n            if (response.status !== 200) {\n              console.warn(`Failed to get file \"${file.fileName}\" from path ${file.path}`);\n              continue;\n            }\n            const fileContent = await response.blob();\n\n            // remove existing file with same name\n            filesOfCurrentTenant = filesOfCurrentTenant.filter(\n              existingFile => existingFile.fileName !== file.fileName\n            );\n            filesToUpload = filesToUpload.filter(\n              existingFile => existingFile.name !== file.fileName\n            );\n\n            filesToUpload.push(new File([fileContent], file.fileName, { type: file.type }));\n          } catch (e) {\n            console.warn(`Failed to add file \"${file.fileName}\" from tenant ${tenantId}`, e);\n          }\n        }\n      } catch (e) {\n        console.warn(`Failed to get asset files from tenant ${tenantId}`);\n      }\n    }\n\n    if (!filesToUpload.length) {\n      return;\n    }\n\n    const newAssets = await this.addFilesToStaticAssets(type, filesToUpload, filesOfCurrentTenant);\n\n    return { oldAssets, newAssets };\n  }\n\n  /**\n   * Adds the given files to the static assets of the given type.\n   */\n  async addFilesToStaticAssets<T extends string = StaticAssetType>(\n    type: T,\n    files: DroppedFile[] | File[],\n    existingFiles: StaticAsset[],\n    throwErrorOnExistingFiles = true\n  ) {\n    const app = await this.getAppForTenant(type);\n    const filesToAddToContents = new Array<StaticAsset>();\n    const addedAt = new Date().toISOString();\n    const filesToUpload: {\n      path: string;\n      contents: Blob;\n    }[] = [];\n    for (const droppedFile of files) {\n      const file = droppedFile instanceof File ? droppedFile : droppedFile.file;\n      const fileName = file.name.toLowerCase().replace(/[^a-zA-Z0-9\\-\\_\\.]/g, '');\n      const newFile = new File([file], fileName, {\n        type: file.type,\n        lastModified: file.lastModified\n      });\n      if (existingFiles.find(existingFile => existingFile.fileName === fileName)) {\n        if (throwErrorOnExistingFiles) {\n          throw Error(`File with name \"${fileName}\" is already existing.`);\n        }\n        existingFiles = existingFiles.filter(existingFile => existingFile.fileName !== fileName);\n      }\n      const fileExtension = file.name.split('.').pop();\n      filesToAddToContents.push({\n        addedAt,\n        lastModified: file.lastModified,\n        extension: fileExtension,\n        fileName: fileName,\n        originalFileName: file.name,\n        path: `/apps/public/${app.contextPath}/${fileName}`,\n        size: newFile.size,\n        type: newFile.type,\n        hashSum: await this.fileService.getHashSumOfFile(newFile)\n      });\n      filesToUpload.push({\n        contents: newFile,\n        path: fileName\n      });\n    }\n\n    const newFiles: StaticAsset[] = [...existingFiles, ...filesToAddToContents];\n    filesToUpload.push({\n      contents: new File([JSON.stringify(newFiles)], this.fileNames.contents, {\n        type: 'application/json'\n      }),\n      path: this.fileNames.contents\n    });\n    await this.appService.binary(app).updateFiles(filesToUpload);\n    await this.ensureAddedFilesArePresent(filesToAddToContents);\n    return newFiles;\n  }\n\n  /**\n   * Gets the static assets app for the given type.\n   */\n  async getAppForTenant<T extends string = StaticAssetType>(type: T): Promise<IApplication> {\n    const contextPath = this.getContextPath(type);\n\n    const { data: manifest } = await this.appService.getManifestOfContextPath(contextPath);\n    if (!manifest.id) {\n      throw Error(`No app with path ${contextPath} found.`);\n    }\n    const { data: app } = await this.appService.detail(manifest.id);\n    if (app.owner?.tenant.id !== this.appstate.currentTenant.value?.name) {\n      throw Error(`No app with path ${contextPath} found that is owned by the current tenant.`);\n    }\n    return app;\n  }\n\n  private getContentJSONForType<T extends string = StaticAssetType>(\n    type: T,\n    appendTenantId?: string | null | undefined\n  ) {\n    const path = this.getPathForContent(type, appendTenantId);\n    return this.getContentJSONFromPath(path);\n  }\n\n  private async getContentJSONFromPath(path: string): Promise<StaticAsset[]> {\n    const response = await this.fetchClient.fetch(path);\n    if (response.status !== 200) {\n      throw Error(`Failed to retrieve ${path}`);\n    }\n\n    const content = await response.json();\n    return content;\n  }\n\n  private async ensureAddedFilesArePresent(files: StaticAsset[]) {\n    for (const file of files) {\n      await this.ensureFileIsPresent(file);\n    }\n  }\n\n  private async ensureFileIsPresent(file: StaticAsset) {\n    for (let i = 0; i < 12; i++) {\n      try {\n        const currentDate = new Date();\n        const response = await this.fetchClient.fetch(\n          `${file.path}?nocache=${currentDate.getTime()}`\n        );\n        if (response.status !== 200) {\n          await this.sleep(5_000);\n          continue;\n        }\n\n        const blob = await response.blob();\n        const hashSum = await this.fileService.getHashSumOfFile(blob);\n        if (hashSum !== file.hashSum) {\n          await this.sleep(5_000);\n          continue;\n        }\n        return;\n      } catch (e) {\n        console.warn(e);\n        continue;\n      }\n    }\n    throw Error(`Unable to retrieve file from ${file.path}`);\n  }\n\n  private sleep(duration: number): Promise<void> {\n    return new Promise<void>(resolve => setTimeout(() => resolve(), duration));\n  }\n\n  private async _listFiles<T extends string = StaticAssetType>(type: T) {\n    try {\n      const files = await this.getContentJSONForType(type);\n      return files;\n    } catch (e) {\n      console.warn(e);\n    }\n\n    return await this.createEmptyApp(type);\n  }\n\n  private getPathForContent<T extends string = StaticAssetType>(\n    type: T,\n    appendTenantId?: string | null | undefined\n  ): string {\n    const currentDate = new Date();\n    const contextPath = this.getContextPath(type, appendTenantId);\n    return `/apps/public/${contextPath}/${\n      this.fileNames.contents\n    }?nocache=${currentDate.getTime()}`;\n  }\n\n  private getContextPath<T extends string = StaticAssetType>(\n    type: T,\n    appendTenantId?: string | false | undefined\n  ) {\n    let contextPath: string = type + '-assets';\n    const tenantId =\n      appendTenantId === undefined ? this.appstate.currentTenant.value?.name : appendTenantId;\n    if (tenantId) {\n      contextPath = `${contextPath}-${tenantId}`;\n    }\n    return contextPath;\n  }\n\n  private async createEmptyApp<T extends string = StaticAssetType>(\n    type: T,\n    appendTenantId?: string | null | undefined\n  ): Promise<StaticAsset[]> {\n    const name = this.getContextPath(type, false);\n    const contextPath = this.getContextPath(type, appendTenantId);\n    const key = `${contextPath}-key`;\n    const manifest: IApplication = {\n      name,\n      contextPath,\n      key,\n      description: `Application containing static assets used for ${type}.`,\n      noAppSwitcher: true,\n      type: ApplicationType.HOSTED,\n      availability: ApplicationAvailability.MARKET\n    };\n\n    const content = [];\n    const zipFile = await this.zip.createZip([\n      {\n        fileName: `${this.fileNames.manifest}`,\n        blob: new Blob([JSON.stringify(manifest)])\n      },\n      {\n        fileName: `${this.fileNames.contents}`,\n        blob: new Blob([JSON.stringify(content)])\n      }\n    ]);\n    const { data: app } = await this.appService.create(manifest);\n    const { data: appBinary } = await this.appService\n      .binary(app)\n      .upload(zipFile, `empty-${name}.zip`);\n    await this.appService.update({ id: app.id, activeVersionId: appBinary.id as string });\n    return [];\n  }\n}\n"]}
@@ -220,17 +220,17 @@ class DataFetchingService {
220
220
  */
221
221
  getLimitExceededMessage(hasNoExportableData, emailDeliverableCount, browserDownloadableCount, nonRetrievableCount, totalDatapointsSelectedForExportCount) {
222
222
  if (hasNoExportableData) {
223
- const message = this.translateService.instant(gettext(`The data for selected datapoint(s) exceed 1,000,000 records, which is the limit for backend processing. To export this data, please reduce the date range.`));
223
+ const message = this.translateService.instant(gettext(`The data for selected data point(s) exceeds 1,000,000 records, which is the limit for backend processing. To export this data, reduce the date range.`));
224
224
  return message;
225
225
  }
226
226
  const message = this.translateService.instant(gettext(`After clicking "Download":<br>
227
227
  <ul>
228
- <li><strong>{{$browserDownloadableCount}}</strong> datapoint(s) exports will be downloaded directly within one file: <em>exported_[csv/excel].zip</em></li>
229
- <li><strong>{{$emailDeliverableCount}}</strong> datapoint(s) exports require further processing. The files will be sent to you via separate emails once completed, which may take some time.</li>
230
- <li><strong>{{$nonRetrievableCount}}</strong> datapoint(s) exports exceeded 1,000,000 records, which is the limit for backend processing. To export these data, please reduce the date range. Otherwise, the data will neither be downloaded nor sent via email.</li>
228
+ <li><strong>{{$browserDownloadableCount}}</strong> data point export(s) will be downloaded directly within one file: <em>exported_[csv/excel].zip</em></li>
229
+ <li><strong>{{$emailDeliverableCount}}</strong> data point export(s) require further processing. The files will be sent to you via separate emails once completed, which may take some time.</li>
230
+ <li><strong>{{$nonRetrievableCount}}</strong> data point export(s) exceeded 1,000,000 records, which is the limit for backend processing. To export these data, reduce the date range. Otherwise, the data will neither be downloaded nor sent via email.</li>
231
231
  </ul>
232
- <p>The total number of data points that can be exported is: <strong>{{$totalExportableDatapointsCount}} out of {{$totalDatapointsSelectedForExportCount}}</strong>.
233
- <p><strong>Note:</strong> The file name convention of files within zip file is: <code>[source]_[fragment_series].[csv/xls]</code></p>`), {
232
+ <p>The total number of data points that can be exported is: <strong>{{$totalExportableDatapointsCount}} out of {{$totalDatapointsSelectedForExportCount}}</strong>.</p>
233
+ <p><strong>Note:</strong> The file name convention of files within a ZIP file is: <code>[source]_[fragment_series].[csv/xls]</code></p>`), {
234
234
  $browserDownloadableCount: browserDownloadableCount,
235
235
  $emailDeliverableCount: emailDeliverableCount,
236
236
  $nonRetrievableCount: nonRetrievableCount,
@@ -726,11 +726,11 @@ class DataPointsExportSelectorPreviewComponent {
726
726
  this.MEASUREMENTS_PREVIEW_ITEMS_LIMIT = MEASUREMENTS_PREVIEW_ITEMS_LIMIT;
727
727
  }
728
728
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: DataPointsExportSelectorPreviewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
729
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: DataPointsExportSelectorPreviewComponent, isStandalone: true, selector: "c8y-datapoints-export-selector-preview", inputs: { hasFetchedDataAnyValuesToExport: "hasFetchedDataAnyValuesToExport", isPreviewLoading: "isPreviewLoading", previewTableData: "previewTableData" }, ngImport: i0, template: "<div class=\"p-t-16 p-l-16 p-r-16 m-b-0\">\n <div class=\"d-flex a-i-center\">\n <label\n class=\"m-b-0 d-flex a-i-center gap-4\"\n [title]=\"'Preview`of exported file`' | translate\"\n >\n {{ 'Preview`of exported file`' | translate }}\n <button\n class=\"btn-help\"\n [attr.aria-label]=\"'Help' | translate\"\n [popover]=\"popoverPreviewTemplate\"\n placement=\"right\"\n triggers=\"focus\"\n container=\"body\"\n type=\"button\"\n [adaptivePosition]=\"true\"\n ></button>\n <ng-template #popoverPreviewTemplate>\n <span translate>\n <p>The preview shows the structure of the raw file from a single source.</p>\n <p>If no data is available, only the column headers are visible.</p>\n <p>\n The preview is limited to\n <b>{{ MEASUREMENTS_PREVIEW_ITEMS_LIMIT }}</b>\n records.\n </p>\n </span>\n </ng-template>\n </label>\n </div>\n <div\n class=\"table-responsive\"\n style=\"min-height: 275px\"\n >\n <table class=\"table\">\n <thead>\n <tr>\n <th>{{ 'Time' | translate }}</th>\n <th>{{ 'Source' | translate }}</th>\n <th>{{ 'Device name' | translate }}</th>\n <th>\n {{ 'Fragment and series' | translate }}\n </th>\n <th>{{ 'Value' | translate }}</th>\n <th>{{ 'Unit' | translate }}</th>\n </tr>\n </thead>\n <ng-container *ngIf=\"hasFetchedDataAnyValuesToExport || isPreviewLoading; else emptyState\">\n <ng-container *ngIf=\"!isPreviewLoading; else loading\">\n <tbody>\n <tr *ngFor=\"let row of previewTableData\">\n <td>{{ row.time }}</td>\n <td>{{ row.source }}</td>\n <td>{{ row.device_name }}</td>\n <td>\n {{ row.fragment_series }}\n </td>\n <td>{{ row.value }}</td>\n <td>{{ row.unit }}</td>\n </tr>\n </tbody>\n </ng-container>\n </ng-container>\n <ng-template #emptyState>\n <tbody>\n <tr>\n <td colspan=\"8\">\n <div class=\"d-col a-i-center\">\n <c8y-ui-empty-state\n [icon]=\"'search'\"\n [title]=\"'No data available.' | translate\"\n [horizontal]=\"true\"\n data-cy=\"datapoints-table-list--empty-state\"\n ></c8y-ui-empty-state>\n </div>\n </td>\n </tr>\n </tbody>\n </ng-template>\n <ng-template #loading>\n <tbody>\n <tr>\n <td colspan=\"8\">\n <c8y-loading></c8y-loading>\n </td>\n </tr>\n </tbody>\n </ng-template>\n </table>\n </div>\n</div>\n", dependencies: [{ kind: "ngmodule", type: A11yModule }, { kind: "ngmodule", type: CoreModule }, { kind: "component", type: i1.EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "pipe", type: i1.C8yTranslatePipe, name: "translate" }, { kind: "directive", type: i1.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i2$1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2$1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i1.LoadingComponent, selector: "c8y-loading", inputs: ["layout", "progress", "message"] }, { kind: "ngmodule", type: PopoverModule }, { kind: "directive", type: i3$1.PopoverDirective, selector: "[popover]", inputs: ["adaptivePosition", "boundariesElement", "popover", "popoverContext", "popoverTitle", "placement", "outsideClick", "triggers", "container", "containerClass", "isOpen", "delay"], outputs: ["onShown", "onHidden"], exportAs: ["bs-popover"] }] }); }
729
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: DataPointsExportSelectorPreviewComponent, isStandalone: true, selector: "c8y-datapoints-export-selector-preview", inputs: { hasFetchedDataAnyValuesToExport: "hasFetchedDataAnyValuesToExport", isPreviewLoading: "isPreviewLoading", previewTableData: "previewTableData" }, ngImport: i0, template: "<div class=\"p-t-16 p-l-16 p-r-16 m-b-0\">\n <div class=\"d-flex a-i-center\">\n <label\n class=\"m-b-0 d-flex a-i-center gap-4\"\n [title]=\"'Preview`of exported file`' | translate\"\n >\n {{ 'Preview`of exported file`' | translate }}\n <button\n class=\"btn-help\"\n [attr.aria-label]=\"'Help' | translate\"\n [popover]=\"popoverPreviewTemplate\"\n placement=\"right\"\n triggers=\"focus\"\n container=\"body\"\n type=\"button\"\n [adaptivePosition]=\"true\"\n ></button>\n <ng-template #popoverPreviewTemplate>\n <p translate>The preview shows the structure of the raw file from a single source.</p>\n <p translate>If no data is available, only the column headers are visible.</p>\n <p\n translate\n [translateParams]=\"{ count: MEASUREMENTS_PREVIEW_ITEMS_LIMIT }\"\n ngNonBindable\n >\n The preview is limited to <b>{{ count }}</b> records.\n </p>\n </ng-template>\n </label>\n </div>\n <div\n class=\"table-responsive\"\n style=\"min-height: 275px\"\n >\n <table class=\"table\">\n <thead>\n <tr>\n <th>{{ 'Time' | translate }}</th>\n <th>{{ 'Source' | translate }}</th>\n <th>{{ 'Device name' | translate }}</th>\n <th>\n {{ 'Fragment and series' | translate }}\n </th>\n <th>{{ 'Value' | translate }}</th>\n <th>{{ 'Unit' | translate }}</th>\n </tr>\n </thead>\n <ng-container *ngIf=\"hasFetchedDataAnyValuesToExport || isPreviewLoading; else emptyState\">\n <ng-container *ngIf=\"!isPreviewLoading; else loading\">\n <tbody>\n <tr *ngFor=\"let row of previewTableData\">\n <td>{{ row.time }}</td>\n <td>{{ row.source }}</td>\n <td>{{ row.device_name }}</td>\n <td>\n {{ row.fragment_series }}\n </td>\n <td>{{ row.value }}</td>\n <td>{{ row.unit }}</td>\n </tr>\n </tbody>\n </ng-container>\n </ng-container>\n <ng-template #emptyState>\n <tbody>\n <tr>\n <td colspan=\"8\">\n <div class=\"d-col a-i-center\">\n <c8y-ui-empty-state\n [icon]=\"'search'\"\n [title]=\"'No data available.' | translate\"\n [horizontal]=\"true\"\n data-cy=\"datapoints-table-list--empty-state\"\n ></c8y-ui-empty-state>\n </div>\n </td>\n </tr>\n </tbody>\n </ng-template>\n <ng-template #loading>\n <tbody>\n <tr>\n <td colspan=\"8\">\n <c8y-loading></c8y-loading>\n </td>\n </tr>\n </tbody>\n </ng-template>\n </table>\n </div>\n</div>\n", dependencies: [{ kind: "ngmodule", type: A11yModule }, { kind: "ngmodule", type: CoreModule }, { kind: "component", type: i1.EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "pipe", type: i1.C8yTranslatePipe, name: "translate" }, { kind: "directive", type: i1.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i2$1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2$1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i1.LoadingComponent, selector: "c8y-loading", inputs: ["layout", "progress", "message"] }, { kind: "ngmodule", type: PopoverModule }, { kind: "directive", type: i3$1.PopoverDirective, selector: "[popover]", inputs: ["adaptivePosition", "boundariesElement", "popover", "popoverContext", "popoverTitle", "placement", "outsideClick", "triggers", "container", "containerClass", "isOpen", "delay"], outputs: ["onShown", "onHidden"], exportAs: ["bs-popover"] }] }); }
730
730
  }
731
731
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: DataPointsExportSelectorPreviewComponent, decorators: [{
732
732
  type: Component,
733
- args: [{ selector: 'c8y-datapoints-export-selector-preview', standalone: true, imports: [A11yModule, CoreModule, PopoverModule], template: "<div class=\"p-t-16 p-l-16 p-r-16 m-b-0\">\n <div class=\"d-flex a-i-center\">\n <label\n class=\"m-b-0 d-flex a-i-center gap-4\"\n [title]=\"'Preview`of exported file`' | translate\"\n >\n {{ 'Preview`of exported file`' | translate }}\n <button\n class=\"btn-help\"\n [attr.aria-label]=\"'Help' | translate\"\n [popover]=\"popoverPreviewTemplate\"\n placement=\"right\"\n triggers=\"focus\"\n container=\"body\"\n type=\"button\"\n [adaptivePosition]=\"true\"\n ></button>\n <ng-template #popoverPreviewTemplate>\n <span translate>\n <p>The preview shows the structure of the raw file from a single source.</p>\n <p>If no data is available, only the column headers are visible.</p>\n <p>\n The preview is limited to\n <b>{{ MEASUREMENTS_PREVIEW_ITEMS_LIMIT }}</b>\n records.\n </p>\n </span>\n </ng-template>\n </label>\n </div>\n <div\n class=\"table-responsive\"\n style=\"min-height: 275px\"\n >\n <table class=\"table\">\n <thead>\n <tr>\n <th>{{ 'Time' | translate }}</th>\n <th>{{ 'Source' | translate }}</th>\n <th>{{ 'Device name' | translate }}</th>\n <th>\n {{ 'Fragment and series' | translate }}\n </th>\n <th>{{ 'Value' | translate }}</th>\n <th>{{ 'Unit' | translate }}</th>\n </tr>\n </thead>\n <ng-container *ngIf=\"hasFetchedDataAnyValuesToExport || isPreviewLoading; else emptyState\">\n <ng-container *ngIf=\"!isPreviewLoading; else loading\">\n <tbody>\n <tr *ngFor=\"let row of previewTableData\">\n <td>{{ row.time }}</td>\n <td>{{ row.source }}</td>\n <td>{{ row.device_name }}</td>\n <td>\n {{ row.fragment_series }}\n </td>\n <td>{{ row.value }}</td>\n <td>{{ row.unit }}</td>\n </tr>\n </tbody>\n </ng-container>\n </ng-container>\n <ng-template #emptyState>\n <tbody>\n <tr>\n <td colspan=\"8\">\n <div class=\"d-col a-i-center\">\n <c8y-ui-empty-state\n [icon]=\"'search'\"\n [title]=\"'No data available.' | translate\"\n [horizontal]=\"true\"\n data-cy=\"datapoints-table-list--empty-state\"\n ></c8y-ui-empty-state>\n </div>\n </td>\n </tr>\n </tbody>\n </ng-template>\n <ng-template #loading>\n <tbody>\n <tr>\n <td colspan=\"8\">\n <c8y-loading></c8y-loading>\n </td>\n </tr>\n </tbody>\n </ng-template>\n </table>\n </div>\n</div>\n" }]
733
+ args: [{ selector: 'c8y-datapoints-export-selector-preview', standalone: true, imports: [A11yModule, CoreModule, PopoverModule], template: "<div class=\"p-t-16 p-l-16 p-r-16 m-b-0\">\n <div class=\"d-flex a-i-center\">\n <label\n class=\"m-b-0 d-flex a-i-center gap-4\"\n [title]=\"'Preview`of exported file`' | translate\"\n >\n {{ 'Preview`of exported file`' | translate }}\n <button\n class=\"btn-help\"\n [attr.aria-label]=\"'Help' | translate\"\n [popover]=\"popoverPreviewTemplate\"\n placement=\"right\"\n triggers=\"focus\"\n container=\"body\"\n type=\"button\"\n [adaptivePosition]=\"true\"\n ></button>\n <ng-template #popoverPreviewTemplate>\n <p translate>The preview shows the structure of the raw file from a single source.</p>\n <p translate>If no data is available, only the column headers are visible.</p>\n <p\n translate\n [translateParams]=\"{ count: MEASUREMENTS_PREVIEW_ITEMS_LIMIT }\"\n ngNonBindable\n >\n The preview is limited to <b>{{ count }}</b> records.\n </p>\n </ng-template>\n </label>\n </div>\n <div\n class=\"table-responsive\"\n style=\"min-height: 275px\"\n >\n <table class=\"table\">\n <thead>\n <tr>\n <th>{{ 'Time' | translate }}</th>\n <th>{{ 'Source' | translate }}</th>\n <th>{{ 'Device name' | translate }}</th>\n <th>\n {{ 'Fragment and series' | translate }}\n </th>\n <th>{{ 'Value' | translate }}</th>\n <th>{{ 'Unit' | translate }}</th>\n </tr>\n </thead>\n <ng-container *ngIf=\"hasFetchedDataAnyValuesToExport || isPreviewLoading; else emptyState\">\n <ng-container *ngIf=\"!isPreviewLoading; else loading\">\n <tbody>\n <tr *ngFor=\"let row of previewTableData\">\n <td>{{ row.time }}</td>\n <td>{{ row.source }}</td>\n <td>{{ row.device_name }}</td>\n <td>\n {{ row.fragment_series }}\n </td>\n <td>{{ row.value }}</td>\n <td>{{ row.unit }}</td>\n </tr>\n </tbody>\n </ng-container>\n </ng-container>\n <ng-template #emptyState>\n <tbody>\n <tr>\n <td colspan=\"8\">\n <div class=\"d-col a-i-center\">\n <c8y-ui-empty-state\n [icon]=\"'search'\"\n [title]=\"'No data available.' | translate\"\n [horizontal]=\"true\"\n data-cy=\"datapoints-table-list--empty-state\"\n ></c8y-ui-empty-state>\n </div>\n </td>\n </tr>\n </tbody>\n </ng-template>\n <ng-template #loading>\n <tbody>\n <tr>\n <td colspan=\"8\">\n <c8y-loading></c8y-loading>\n </td>\n </tr>\n </tbody>\n </ng-template>\n </table>\n </div>\n</div>\n" }]
734
734
  }], propDecorators: { hasFetchedDataAnyValuesToExport: [{
735
735
  type: Input
736
736
  }], isPreviewLoading: [{