UNPKG

@c8y/ngx-components

Version:

Angular modules for Cumulocity IoT applications

238 lines 44.6 kB
import { Component, EventEmitter, Input, Output } from '@angular/core'; import { InventoryBinaryService, InventoryService } from '@c8y/client'; import { AlertService, AssetTypesRealtimeService, gettext } from '@c8y/ngx-components'; import { sortBy, toPairs, fromPairs, find } from 'lodash-es'; import { firstValueFrom } from 'rxjs'; import * as i0 from "@angular/core"; import * as i1 from "@c8y/ngx-components"; import * as i2 from "@c8y/client"; import * as i3 from "@angular/common"; import * as i4 from "ngx-bootstrap/tooltip"; import * as i5 from "./asset-properties-item.component"; import * as i6 from "./location/asset-location.component"; export class AssetPropertiesComponent { constructor(assetTypes, inventory, inventoryBinary, alert) { this.assetTypes = assetTypes; this.inventory = inventory; this.inventoryBinary = inventoryBinary; this.alert = alert; this.assetChange = new EventEmitter(); this.properties = []; this.customProperties = []; this.isEdit = false; this.isLoading = false; } ngOnChanges(changes) { if (changes.asset) { // Back button handling, as component is not destroyed this.assetType = undefined; this.customProperties = []; this.loadAsset(); } } async loadAsset() { this.isLoading = true; const assetType$ = this.assetTypes.getAssetTypeByName$(this.asset.type); this.assetType = await firstValueFrom(assetType$); try { this.properties = this.keepOrder(this.assetType?.c8y_IsAssetType?.properties, this.properties); } catch (ex) { console.warn(ex); } this.customProperties = await this.resolveCustomProperties(this.properties); this.isLoading = false; } async resolveCustomProperties(managedObjects) { const properties = []; for (const mo of managedObjects) { if (mo.c8y_JsonSchema) { const [item] = await this.parseItem(mo, mo.c8y_JsonSchema.properties, this.asset); this.setItemRequired(item, mo); properties.push(item); } } return properties; } deleteTitleFromMOJsonSchema(mo) { const schemaProperties = mo?.c8y_JsonSchema?.properties; const property = Object.keys(schemaProperties || {})[0]; delete (mo?.c8y_JsonSchema?.properties[property] || {}).title; } /** * This method is used to order the complex properties in the order specified by the user in asset properties screen. * @param mo - Managed object of the complex property associated with the asset. */ orderComplexProperties(mo) { const complexProperties = mo.c8y_JsonSchema.properties[mo.name]?.['properties']; const keyValuesArray = toPairs(complexProperties); const orderedProperties = sortBy(keyValuesArray, ([, value]) => value.order); mo.c8y_JsonSchema.properties[mo.name]['properties'] = fromPairs(orderedProperties); } async parseItem(mo, properties, asset) { if (!asset) { return []; } const keys = Object.keys(properties); const items = []; for (const key of keys) { const type = properties[key].type; const title = properties[key].title; let value = this.getTypeValue(type, asset[key]); let file; if (type === 'file' && value) { const fileId = typeof value === 'object' ? value[0]?.file?.id : value; const fileData = await this.getFileManagedObject(fileId); file = fileData; value = [fileData]; } else if (type === 'date') { const valueDate = new Date(value); value = !isNaN(valueDate.getTime()) ? valueDate : ''; } if (type === 'object') { // remove title to avoid excessive property name on asset complex properties form this.deleteTitleFromMOJsonSchema(mo); this.orderComplexProperties(mo); if (!value) { value = {}; for (const prop in properties[key].properties) { value[prop] = this.getTypeValue(properties[key].properties[prop].type, null); } } } items.push({ key, value, label: title || mo.label, type, description: mo.description, file, complex: type === 'object' ? await this.parseItem(mo, properties[key].properties, value) : undefined, isEdit: false, jsonSchema: mo.c8y_JsonSchema }); } return items; } toggleEdit(prop) { prop.isEdit = !prop.isEdit; } async getFileManagedObject(id) { try { const { data } = await this.inventory.detail(id); return data; } catch (ex) { this.alert.addServerFailure(ex); } } async save(propertyValue, prop) { try { if (prop.type === 'object') { this.updateUndefinedToPropTypeValue(prop, propertyValue[prop.key]); } else { this.updateUndefinedToPropTypeValue(prop, propertyValue); } propertyValue = await this.uploadFiles(propertyValue, prop.value); // Avoid making a PUT request containing just the id, as response body might be incomplete const hasValues = Object.values(propertyValue).some(value => value !== undefined); if (!hasValues) { this.toggleEdit(prop); return; } const updatedAsset = { id: this.asset.id, ...propertyValue }; const { data } = await this.inventory.update(updatedAsset); this.toggleEdit(prop); this.asset = data; this.assetChange.emit(this.asset); await this.loadAsset(); this.alert.success(gettext('Asset properties updated.')); } catch (ex) { this.alert.addServerFailure(ex); this.toggleEdit(prop); } } updateUndefinedToPropTypeValue(prop, propertyValue) { for (const [key, value] of Object.entries(propertyValue)) { const property = prop.complex ? find(prop.complex, { key: key }) : prop; propertyValue[key] = this.getTypeValue(property.type, value); } } getTypeValue(propType, value) { if (value || (propType === 'boolean' && value !== undefined)) return value; switch (propType) { case 'number': case 'boolean': return value || value === 0 ? value : null; default: return ''; } } keepOrder(correctOrderedIds, properties) { const orderedProperties = correctOrderedIds.map(({ id }) => { const foundProperty = properties.find(property => property.id === id); if (!foundProperty) { throw new Error('Custom property mismatch'); } return foundProperty; }); return orderedProperties; } async uploadFiles(model, moId) { const keys = Object.keys(model); for (const key of keys) { if (Array.isArray(model[key]) && model[key][0]?.file instanceof File) { try { const upload = await this.inventoryBinary.create(model[key][0].file); try { if (moId && moId[0]) { await this.inventory.childAdditionsRemove(moId[0], this.asset.id); } } catch (ex) { throw ex; } model[key] = upload.data.id; await this.inventory.childAdditionsAdd(upload.data.id, this.asset.id); } catch (ex) { throw ex; } } } return model; } setItemRequired(item, mo) { const isAssetPropertyRequired = !!this.assetType?.c8y_IsAssetType?.properties.find(p => p.id === mo.id)?.isRequired; if (!isAssetPropertyRequired) { return; } const isComplexProperty = !!item?.complex?.length; if (isComplexProperty) { const complexProperty = item.jsonSchema?.properties?.[mo.c8y_JsonSchema.key]; complexProperty.required = item.complex.map(({ key }) => key); } else { item.jsonSchema.required = [mo.c8y_JsonSchema.key]; } } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AssetPropertiesComponent, deps: [{ token: i1.AssetTypesRealtimeService }, { token: i2.InventoryService }, { token: i2.InventoryBinaryService }, { token: i1.AlertService }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: AssetPropertiesComponent, selector: "c8y-asset-properties", inputs: { asset: "asset", properties: "properties" }, outputs: { assetChange: "assetChange" }, usesOnChanges: true, ngImport: i0, template: "<ng-container>\n <div class=\"card-header bg-inherit separator sticky-top\">\n <h1\n class=\"card-title p-t-4 p-b-4\"\n ngNonBindable\n translate\n [translateParams]=\"{ label: assetType?.label || '' | translate }\"\n >\n {{ label }} properties\n </h1>\n </div>\n <div class=\"card-block\">\n <div\n class=\"text-center\"\n *ngIf=\"isLoading\"\n >\n <c8y-loading></c8y-loading>\n </div>\n\n <ng-container *ngIf=\"!isLoading\">\n <div\n class=\"card m-b-8\"\n title=\"{{ prop.description | translate }}\"\n *ngFor=\"let prop of customProperties\"\n [ngClass]=\"{ 'card-highlight': prop.isEdit }\"\n >\n <div\n class=\"card-block\"\n [ngClass]=\"{ 'p-b-0': prop.isEdit }\"\n >\n <div\n class=\"d-flex p-b-8 a-i-center\"\n *ngIf=\"!prop.isEdit\"\n >\n <p\n class=\"text-medium text-truncate\"\n title=\"{{ prop?.label | translate }}\"\n >\n {{ prop?.label | translate }}\n </p>\n <button\n class=\"btn btn-dot m-l-auto text-12\"\n [attr.aria-label]=\"'Edit' | translate\"\n tooltip=\"{{ 'Edit' | translate }}\"\n type=\"button\"\n [delay]=\"500\"\n (click)=\"toggleEdit(prop)\"\n >\n <i c8yIcon=\"pencil\"></i>\n </button>\n </div>\n <c8y-asset-properties-item\n #assetProps\n [file]=\"prop.file\"\n [key]=\"prop.key\"\n [type]=\"prop.type\"\n [value]=\"prop.value\"\n [complex]=\"prop.complex\"\n [isEdit]=\"prop.isEdit\"\n [jsonSchema]=\"prop.jsonSchema\"\n ></c8y-asset-properties-item>\n <div *ngIf=\"prop.key === 'c8y_Position'\">\n <c8y-asset-location\n [locationMO]=\"asset\"\n [isEdit]=\"prop.isEdit\"\n [form]=\"assetProps.form\"\n ></c8y-asset-location>\n </div>\n </div>\n <div\n class=\"card-footer p-t-0\"\n *ngIf=\"prop.isEdit\"\n >\n <button\n class=\"btn btn-default btn-sm\"\n title=\"{{ 'Cancel' | translate }}\"\n type=\"button\"\n (click)=\"toggleEdit(prop)\"\n >\n {{ 'Cancel' | translate }}\n </button>\n <button\n class=\"btn btn-primary btn-sm\"\n title=\"{{ 'Save' | translate }}\"\n type=\"button\"\n [disabled]=\"!assetProps?.form?.valid || !assetProps?.form?.dirty \"\n (click)=\"save(assetProps.form.value, prop)\"\n >\n {{ 'Save' | translate }}\n </button>\n </div>\n </div>\n </ng-container>\n </div>\n</ng-container>\n", dependencies: [{ kind: "directive", type: i1.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i1.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i1.LoadingComponent, selector: "c8y-loading", inputs: ["layout", "progress", "message"] }, { kind: "directive", type: i4.TooltipDirective, selector: "[tooltip], [tooltipHtml]", inputs: ["adaptivePosition", "tooltip", "placement", "triggers", "container", "containerClass", "boundariesElement", "isOpen", "isDisabled", "delay", "tooltipHtml", "tooltipPlacement", "tooltipIsOpen", "tooltipEnable", "tooltipAppendToBody", "tooltipAnimation", "tooltipClass", "tooltipContext", "tooltipPopupDelay", "tooltipFadeDuration", "tooltipTrigger"], outputs: ["tooltipChange", "onShown", "onHidden", "tooltipStateChanged"], exportAs: ["bs-tooltip"] }, { kind: "component", type: i5.AssetPropertiesItemComponent, selector: "c8y-asset-properties-item", inputs: ["key", "value", "label", "type", "file", "complex", "isEdit", "jsonSchema"] }, { kind: "component", type: i6.AssetLocationComponent, selector: "c8y-asset-location", inputs: ["isEdit", "locationMO", "form"] }, { kind: "pipe", type: i1.C8yTranslatePipe, name: "translate" }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AssetPropertiesComponent, decorators: [{ type: Component, args: [{ selector: 'c8y-asset-properties', template: "<ng-container>\n <div class=\"card-header bg-inherit separator sticky-top\">\n <h1\n class=\"card-title p-t-4 p-b-4\"\n ngNonBindable\n translate\n [translateParams]=\"{ label: assetType?.label || '' | translate }\"\n >\n {{ label }} properties\n </h1>\n </div>\n <div class=\"card-block\">\n <div\n class=\"text-center\"\n *ngIf=\"isLoading\"\n >\n <c8y-loading></c8y-loading>\n </div>\n\n <ng-container *ngIf=\"!isLoading\">\n <div\n class=\"card m-b-8\"\n title=\"{{ prop.description | translate }}\"\n *ngFor=\"let prop of customProperties\"\n [ngClass]=\"{ 'card-highlight': prop.isEdit }\"\n >\n <div\n class=\"card-block\"\n [ngClass]=\"{ 'p-b-0': prop.isEdit }\"\n >\n <div\n class=\"d-flex p-b-8 a-i-center\"\n *ngIf=\"!prop.isEdit\"\n >\n <p\n class=\"text-medium text-truncate\"\n title=\"{{ prop?.label | translate }}\"\n >\n {{ prop?.label | translate }}\n </p>\n <button\n class=\"btn btn-dot m-l-auto text-12\"\n [attr.aria-label]=\"'Edit' | translate\"\n tooltip=\"{{ 'Edit' | translate }}\"\n type=\"button\"\n [delay]=\"500\"\n (click)=\"toggleEdit(prop)\"\n >\n <i c8yIcon=\"pencil\"></i>\n </button>\n </div>\n <c8y-asset-properties-item\n #assetProps\n [file]=\"prop.file\"\n [key]=\"prop.key\"\n [type]=\"prop.type\"\n [value]=\"prop.value\"\n [complex]=\"prop.complex\"\n [isEdit]=\"prop.isEdit\"\n [jsonSchema]=\"prop.jsonSchema\"\n ></c8y-asset-properties-item>\n <div *ngIf=\"prop.key === 'c8y_Position'\">\n <c8y-asset-location\n [locationMO]=\"asset\"\n [isEdit]=\"prop.isEdit\"\n [form]=\"assetProps.form\"\n ></c8y-asset-location>\n </div>\n </div>\n <div\n class=\"card-footer p-t-0\"\n *ngIf=\"prop.isEdit\"\n >\n <button\n class=\"btn btn-default btn-sm\"\n title=\"{{ 'Cancel' | translate }}\"\n type=\"button\"\n (click)=\"toggleEdit(prop)\"\n >\n {{ 'Cancel' | translate }}\n </button>\n <button\n class=\"btn btn-primary btn-sm\"\n title=\"{{ 'Save' | translate }}\"\n type=\"button\"\n [disabled]=\"!assetProps?.form?.valid || !assetProps?.form?.dirty \"\n (click)=\"save(assetProps.form.value, prop)\"\n >\n {{ 'Save' | translate }}\n </button>\n </div>\n </div>\n </ng-container>\n </div>\n</ng-container>\n" }] }], ctorParameters: () => [{ type: i1.AssetTypesRealtimeService }, { type: i2.InventoryService }, { type: i2.InventoryBinaryService }, { type: i1.AlertService }], propDecorators: { asset: [{ type: Input }], assetChange: [{ type: Output }], properties: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"asset-properties.component.js","sourceRoot":"","sources":["../../../sub-assets/asset-properties.component.ts","../../../sub-assets/asset-properties.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnChanges, Output, SimpleChanges } from '@angular/core';\nimport {\n  IManagedObject,\n  IManagedObjectBinary,\n  InventoryBinaryService,\n  InventoryService\n} from '@c8y/client';\nimport { AlertService, AssetTypesRealtimeService, gettext } from '@c8y/ngx-components';\nimport { AssetPropertiesItem } from './asset-properties.model';\nimport { JSONSchema7 } from 'json-schema';\nimport { sortBy, toPairs, fromPairs, find } from 'lodash-es';\nimport { firstValueFrom } from 'rxjs';\n\n@Component({\n  selector: 'c8y-asset-properties',\n  templateUrl: './asset-properties.component.html'\n})\nexport class AssetPropertiesComponent implements OnChanges {\n  @Input() asset: IManagedObject;\n  @Output() assetChange = new EventEmitter<IManagedObject>();\n\n  @Input()\n  properties: IManagedObject[] = [];\n\n  assetType: IManagedObject;\n  customProperties: AssetPropertiesItem[] = [];\n  isEdit = false;\n  isLoading = false;\n\n  constructor(\n    private assetTypes: AssetTypesRealtimeService,\n    private inventory: InventoryService,\n    private inventoryBinary: InventoryBinaryService,\n    private alert: AlertService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.asset) {\n      // Back button handling, as component is not destroyed\n      this.assetType = undefined;\n      this.customProperties = [];\n      this.loadAsset();\n    }\n  }\n\n  async loadAsset() {\n    this.isLoading = true;\n    const assetType$ = this.assetTypes.getAssetTypeByName$(this.asset.type);\n    this.assetType = await firstValueFrom(assetType$);\n    try {\n      this.properties = this.keepOrder(\n        this.assetType?.c8y_IsAssetType?.properties,\n        this.properties\n      );\n    } catch (ex) {\n      console.warn(ex);\n    }\n    this.customProperties = await this.resolveCustomProperties(this.properties);\n    this.isLoading = false;\n  }\n\n  async resolveCustomProperties(managedObjects: IManagedObject[]) {\n    const properties = [];\n    for (const mo of managedObjects) {\n      if (mo.c8y_JsonSchema) {\n        const [item] = await this.parseItem(mo, mo.c8y_JsonSchema.properties, this.asset);\n        this.setItemRequired(item, mo);\n        properties.push(item);\n      }\n    }\n    return properties;\n  }\n\n  deleteTitleFromMOJsonSchema(mo: IManagedObject) {\n    const schemaProperties = mo?.c8y_JsonSchema?.properties;\n    const property = Object.keys(schemaProperties || {})[0];\n    delete (mo?.c8y_JsonSchema?.properties[property] || {}).title;\n  }\n\n  /**\n   * This method is used to order the complex properties in the order specified by the user in asset properties screen.\n   * @param mo - Managed object of the complex property associated with the asset.\n   */\n  orderComplexProperties(mo: IManagedObject) {\n    const complexProperties = mo.c8y_JsonSchema.properties[mo.name]?.['properties'];\n    const keyValuesArray = toPairs(complexProperties);\n    const orderedProperties = sortBy(keyValuesArray, ([, value]) => value.order);\n    mo.c8y_JsonSchema.properties[mo.name]['properties'] = fromPairs(orderedProperties);\n  }\n\n  async parseItem(mo: IManagedObject, properties, asset): Promise<AssetPropertiesItem[]> {\n    if (!asset) {\n      return [];\n    }\n    const keys = Object.keys(properties);\n    const items: AssetPropertiesItem[] = [];\n\n    for (const key of keys) {\n      const type = properties[key].type;\n      const title = properties[key].title;\n      let value = this.getTypeValue(type, asset[key]);\n      let file;\n      if (type === 'file' && value) {\n        const fileId = typeof value === 'object' ? value[0]?.file?.id : value;\n        const fileData = await this.getFileManagedObject(fileId);\n        file = fileData;\n        value = [fileData];\n      } else if (type === 'date') {\n        const valueDate = new Date(value);\n        value = !isNaN(valueDate.getTime()) ? valueDate : '';\n      }\n      if (type === 'object') {\n        // remove title to avoid excessive property name on asset complex properties form\n        this.deleteTitleFromMOJsonSchema(mo);\n        this.orderComplexProperties(mo);\n        if (!value) {\n          value = {};\n          for (const prop in properties[key].properties) {\n            value[prop] = this.getTypeValue(properties[key].properties[prop].type, null);\n          }\n        }\n      }\n      items.push({\n        key,\n        value,\n        label: title || mo.label,\n        type,\n        description: mo.description,\n        file,\n        complex:\n          type === 'object'\n            ? await this.parseItem(mo, properties[key].properties, value)\n            : undefined,\n        isEdit: false,\n        jsonSchema: mo.c8y_JsonSchema\n      });\n    }\n    return items;\n  }\n\n  toggleEdit(prop: AssetPropertiesItem) {\n    prop.isEdit = !prop.isEdit;\n  }\n\n  async getFileManagedObject(id: string) {\n    try {\n      const { data } = await this.inventory.detail(id);\n      return data;\n    } catch (ex) {\n      this.alert.addServerFailure(ex);\n    }\n  }\n\n  async save(propertyValue, prop: AssetPropertiesItem): Promise<void> {\n    try {\n      if (prop.type === 'object') {\n        this.updateUndefinedToPropTypeValue(prop, propertyValue[prop.key]);\n      } else {\n        this.updateUndefinedToPropTypeValue(prop, propertyValue);\n      }\n\n      propertyValue = await this.uploadFiles(propertyValue, prop.value);\n\n      // Avoid making a PUT request containing just the id, as response body might be incomplete\n      const hasValues = Object.values(propertyValue).some(value => value !== undefined);\n      if (!hasValues) {\n        this.toggleEdit(prop);\n        return;\n      }\n      const updatedAsset = { id: this.asset.id, ...propertyValue };\n      const { data } = await this.inventory.update(updatedAsset);\n      this.toggleEdit(prop);\n      this.asset = data;\n      this.assetChange.emit(this.asset);\n      await this.loadAsset();\n      this.alert.success(gettext('Asset properties updated.'));\n    } catch (ex) {\n      this.alert.addServerFailure(ex);\n      this.toggleEdit(prop);\n    }\n  }\n\n  private updateUndefinedToPropTypeValue(prop, propertyValue) {\n    for (const [key, value] of Object.entries(propertyValue)) {\n      const property = prop.complex ? find(prop.complex, { key: key }) : prop;\n      propertyValue[key] = this.getTypeValue(property.type, value);\n    }\n  }\n\n  private getTypeValue(propType: string, value) {\n    if (value || (propType === 'boolean' && value !== undefined)) return value;\n\n    switch (propType) {\n      case 'number':\n      case 'boolean':\n        return value || value === 0 ? value : null;\n\n      default:\n        return '';\n    }\n  }\n\n  private keepOrder(correctOrderedIds, properties) {\n    const orderedProperties = correctOrderedIds.map(({ id }) => {\n      const foundProperty = properties.find(property => property.id === id);\n      if (!foundProperty) {\n        throw new Error('Custom property mismatch');\n      }\n      return foundProperty;\n    });\n    return orderedProperties;\n  }\n\n  private async uploadFiles(model: object, moId?: IManagedObjectBinary[]): Promise<object> {\n    const keys = Object.keys(model);\n    for (const key of keys) {\n      if (Array.isArray(model[key]) && model[key][0]?.file instanceof File) {\n        try {\n          const upload = await this.inventoryBinary.create(model[key][0].file);\n          try {\n            if (moId && moId[0]) {\n              await this.inventory.childAdditionsRemove(moId[0], this.asset.id);\n            }\n          } catch (ex) {\n            throw ex;\n          }\n          model[key] = upload.data.id;\n          await this.inventory.childAdditionsAdd(upload.data.id, this.asset.id);\n        } catch (ex) {\n          throw ex;\n        }\n      }\n    }\n    return model;\n  }\n\n  private setItemRequired(item: AssetPropertiesItem, mo: IManagedObject): void {\n    const isAssetPropertyRequired = !!this.assetType?.c8y_IsAssetType?.properties.find(\n      p => p.id === mo.id\n    )?.isRequired;\n    if (!isAssetPropertyRequired) {\n      return;\n    }\n    const isComplexProperty = !!item?.complex?.length;\n    if (isComplexProperty) {\n      const complexProperty = item.jsonSchema?.properties?.[mo.c8y_JsonSchema.key] as JSONSchema7;\n      complexProperty.required = item.complex.map(({ key }) => key);\n    } else {\n      item.jsonSchema.required = [mo.c8y_JsonSchema.key];\n    }\n  }\n}\n","<ng-container>\n  <div class=\"card-header bg-inherit separator sticky-top\">\n    <h1\n      class=\"card-title p-t-4 p-b-4\"\n      ngNonBindable\n      translate\n      [translateParams]=\"{ label: assetType?.label || '' | translate }\"\n    >\n      {{ label }} properties\n    </h1>\n  </div>\n  <div class=\"card-block\">\n    <div\n      class=\"text-center\"\n      *ngIf=\"isLoading\"\n    >\n      <c8y-loading></c8y-loading>\n    </div>\n\n    <ng-container *ngIf=\"!isLoading\">\n      <div\n        class=\"card m-b-8\"\n        title=\"{{ prop.description | translate }}\"\n        *ngFor=\"let prop of customProperties\"\n        [ngClass]=\"{ 'card-highlight': prop.isEdit }\"\n      >\n        <div\n          class=\"card-block\"\n          [ngClass]=\"{ 'p-b-0': prop.isEdit }\"\n        >\n          <div\n            class=\"d-flex p-b-8 a-i-center\"\n            *ngIf=\"!prop.isEdit\"\n          >\n            <p\n              class=\"text-medium text-truncate\"\n              title=\"{{ prop?.label | translate }}\"\n            >\n              {{ prop?.label | translate }}\n            </p>\n            <button\n              class=\"btn btn-dot m-l-auto text-12\"\n              [attr.aria-label]=\"'Edit' | translate\"\n              tooltip=\"{{ 'Edit' | translate }}\"\n              type=\"button\"\n              [delay]=\"500\"\n              (click)=\"toggleEdit(prop)\"\n            >\n              <i c8yIcon=\"pencil\"></i>\n            </button>\n          </div>\n          <c8y-asset-properties-item\n            #assetProps\n            [file]=\"prop.file\"\n            [key]=\"prop.key\"\n            [type]=\"prop.type\"\n            [value]=\"prop.value\"\n            [complex]=\"prop.complex\"\n            [isEdit]=\"prop.isEdit\"\n            [jsonSchema]=\"prop.jsonSchema\"\n          ></c8y-asset-properties-item>\n          <div *ngIf=\"prop.key === 'c8y_Position'\">\n            <c8y-asset-location\n              [locationMO]=\"asset\"\n              [isEdit]=\"prop.isEdit\"\n              [form]=\"assetProps.form\"\n            ></c8y-asset-location>\n          </div>\n        </div>\n        <div\n          class=\"card-footer p-t-0\"\n          *ngIf=\"prop.isEdit\"\n        >\n          <button\n            class=\"btn btn-default btn-sm\"\n            title=\"{{ 'Cancel' | translate }}\"\n            type=\"button\"\n            (click)=\"toggleEdit(prop)\"\n          >\n            {{ 'Cancel' | translate }}\n          </button>\n          <button\n            class=\"btn btn-primary btn-sm\"\n            title=\"{{ 'Save' | translate }}\"\n            type=\"button\"\n            [disabled]=\"!assetProps?.form?.valid || !assetProps?.form?.dirty \"\n            (click)=\"save(assetProps.form.value, prop)\"\n          >\n            {{ 'Save' | translate }}\n          </button>\n        </div>\n      </div>\n    </ng-container>\n  </div>\n</ng-container>\n"]}