UNPKG

@iotize/ionic

Version:

Iotize specific building blocks on top of @ionic/angular.

243 lines 39.2 kB
import { ChangeDetectorRef, Component, EventEmitter, Input, NgZone, Output, } from '@angular/core'; import { AlertController, ModalController } from '@ionic/angular'; import { isCodeError } from '@iotize/common/error'; import { TapClientError } from '@iotize/tap/client/impl'; import { BehaviorSubject, merge, of, Subject } from 'rxjs'; import { filter, shareReplay, startWith, switchMap, tap } from 'rxjs/operators'; import { CurrentDeviceService } from '../current-device.service'; import { LibError } from '../errors'; import { isNfcTapRequiredError } from '../nfc/utility'; import { PendingCallManager } from '../pending-call-manager'; import { TapValueEditorModalComponent } from '../tap-value-editor-modal/tap-value-editor-modal.component'; import * as i0 from "@angular/core"; import * as i1 from "@ionic/angular"; import * as i2 from "../current-device.service"; import * as i3 from "@angular/common"; function isTapNotConnectedError(err) { return (isNfcTapRequiredError(err) || isCodeError(TapClientError.Code.NotConnectedError, err)); } export class TapValueEditorContainerComponent { set variable(v) { this.variableChange.next(v); } /** * Force value to display */ set value(v) { this.forceValueChange.next(v); } constructor(alertDialog, tapService, changeDetectorRef, ngZone, modalController) { this.alertDialog = alertDialog; this.tapService = tapService; this.changeDetectorRef = changeDetectorRef; this.ngZone = ngZone; this.modalController = modalController; this.button = false; this.inputOptions = { type: 'text', }; /** * Show refresh button */ this.showRefreshButton = false; /** * Show submit button */ this.showSubmitButton = false; /** * Show edit button */ this.showEditButton = false; /** * Enable edition mode */ this.editModeChange = new BehaviorSubject(false); this.loadingChange = new BehaviorSubject(false); this.lines = 'none'; this.modalEdition = false; this.submit = new EventEmitter(); this.refresh = new EventEmitter(); this.variableChange = new BehaviorSubject(undefined); this.variableValuesStream = this.variableChange.pipe(filter((v) => !!v), switchMap((v) => v.values), tap((v) => { this.lastReadValue = v; })); this.forceValueChange = new Subject(); this.valueToDisplay = this.editModeChange.pipe(switchMap((editMode) => { if (editMode || this.pendingSubmitValue !== undefined) { return of(this.valueToSubmit); } else { const lastValue = this.variableChange.value?.valueSnapshot; return merge(this.variableValuesStream.pipe(startWith(lastValue)), this.forceValueChange); } }), shareReplay(1)); this.destroyed = new Subject(); } ngOnInit() { this.pendingCallManager = PendingCallManager.create(this.tapService); this.pendingCallManager.pendingCallResult.subscribe((error) => { this.pendingSubmitValue = undefined; this.valueToSubmit = undefined; this.editModeChange.next(false); this.error = error; this.changeDetectorRef.detectChanges(); }); } // hasValueChanged(value: any) { // return this._lastValueFromDevice !== value; // } submitValue() { if (this.valueToSubmit === undefined) { console.warn('No value to submit yet'); return; } this._submitValue(this.valueToSubmit); } async notifyValueChange(newValue, forceWrite = false) { if (forceWrite || this.lastReadValue !== newValue) { await this._submitValue(newValue); } } async openErrorDialog(error) { const dialog = await this.alertDialog.create({ header: 'Error', message: error.message, buttons: ['Ok'], }); await dialog.present(); } async explainWaitForSubmit() { const alertDialog = await this.alertDialog.create({ header: 'Value not sent yet', message: 'New value will be sent to the device once you are reconnected', buttons: ['Ok'], }); await alertDialog.present(); } async _submitValue(newValue) { this.submit.emit(newValue); // OLD System const variable = this.variableChange.value; if (!variable) { this.error = LibError.componentArgumentRequired(this.constructor.name, 'variable'); return; } if (!this.pendingCallManager) { return; } console.log(`[${variable.id}]`, 'Submit value', newValue); await this.startAction(this.pendingCallManager .exec(async () => { const result = await variable.write(newValue); this.editModeChange.next(false); this.forceValueChange.next(newValue); this.error = undefined; this.pendingSubmitValue = undefined; return result; }) .catch((err) => { if (!isTapNotConnectedError(err)) { console.warn(`[${variable.id}]`, `onSubmit value "${newValue}" error`, err); this.error = err; } else { this.pendingSubmitValue = newValue; this.valueToSubmit = newValue; this.editModeChange.next(false); } })); } async refreshValue() { this.refresh.next(undefined); const variable = this.variableChange.value; if (variable) { await this.startAction(variable .read() .then((newValue) => { this.forceValueChange.next(newValue); }) .catch((err) => { console.log(err); })); } } async enableEditMode() { this.modalEdition ? this.showEditModal() : this.editModeChange.next(true); } async showEditModal() { const lastValue = this.variableChange.value?.valueSnapshot; const modal = await this.modalController.create({ component: TapValueEditorModalComponent, componentProps: { inputOptions: this.inputOptions, value: lastValue, }, }); await modal.present(); const result = await modal.onDidDismiss(); if (result) { if (result.role === 'submit') { const newValue = result.data.value; await this.notifyValueChange(newValue); } } } async startAction(actions) { this.ngZone.run(async () => { this.loadingChange.next(true); try { return await actions; } catch (err) { console.warn('action failed', err); } finally { this.loadingChange.next(false); } }); } ngOnDestroy() { this.destroyed.next(); this.destroyed.complete(); this.pendingCallManager?.destroy(); } cancelEdit() { this.pendingCallManager?.cancel(); this.error = undefined; this.pendingSubmitValue = undefined; this.valueToSubmit = undefined; this.editModeChange.next(false); } } /** @nocollapse */ TapValueEditorContainerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TapValueEditorContainerComponent, deps: [{ token: i1.AlertController }, { token: i2.CurrentDeviceService }, { token: i0.ChangeDetectorRef }, { token: i0.NgZone }, { token: i1.ModalController }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ TapValueEditorContainerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: TapValueEditorContainerComponent, selector: "tap-value-editor-container", inputs: { button: "button", variable: "variable", inputOptions: "inputOptions", showRefreshButton: "showRefreshButton", showSubmitButton: "showSubmitButton", showEditButton: "showEditButton", error: "error", lines: "lines", modalEdition: "modalEdition", value: "value" }, outputs: { submit: "submit", refresh: "refresh" }, ngImport: i0, template: "<ion-item\n style=\"pointer-events: auto\"\n [lines]=\"lines\"\n [button]=\"button\"\n [ngClass]=\"{ 'item-input': editModeChange.value }\"\n>\n <ng-content></ng-content>\n <ion-buttons\n slot=\"end\"\n *ngIf=\"\n showEditButton ||\n showRefreshButton ||\n error ||\n pendingSubmitValue !== undefined\n \"\n >\n <ion-button\n (click)=\"explainWaitForSubmit(); $event.stopPropagation()\"\n *ngIf=\"pendingSubmitValue !== undefined\"\n [disabled]=\"false\"\n color=\"warning\"\n >\n <ion-icon name=\"alert-circle\"></ion-icon>\n </ion-button>\n <ion-button\n (click)=\"enableEditMode(); $event.stopPropagation()\"\n class=\"btn-enable-edit-mode\"\n *ngIf=\"\n pendingSubmitValue === undefined &&\n showEditButton &&\n !editModeChange.value\n \"\n >\n <ion-icon name=\"create\"></ion-icon>\n </ion-button>\n <ion-button\n (click)=\"submitValue(); $event.stopPropagation()\"\n *ngIf=\"\n showSubmitButton &&\n editModeChange.value &&\n pendingSubmitValue === undefined\n \"\n [disabled]=\"loadingChange | async\"\n class=\"btn-submit-value\"\n >\n <ion-icon name=\"send\"></ion-icon>\n </ion-button>\n <ion-button\n (click)=\"cancelEdit(); $event.stopPropagation()\"\n *ngIf=\"editModeChange.value || pendingSubmitValue !== undefined\"\n [disabled]=\"loadingChange | async\"\n class=\"btn-close-edit-mode\"\n >\n <ion-icon name=\"close\"></ion-icon>\n </ion-button>\n <ion-button\n *ngIf=\"showRefreshButton\"\n (click)=\"refreshValue(); $event.stopPropagation()\"\n [disabled]=\"loadingChange | async\"\n class=\"btn-refresh-value\"\n >\n <ion-icon name=\"sync\"></ion-icon>\n </ion-button>\n <ion-button\n *ngIf=\"error\"\n (click)=\"openErrorDialog(error); $event.stopPropagation()\"\n class=\"btn-show-error\"\n >\n <ion-icon slot=\"end\" name=\"alert-circle\" color=\"danger\"></ion-icon>\n </ion-button>\n </ion-buttons>\n <ion-spinner *ngIf=\"loadingChange | async\" slot=\"end\"></ion-spinner>\n</ion-item>\n", styles: ["ion-buttons[slot=end]{margin-inline-start:6px}ion-item{--ion-item-background: transparent;--color: initial}\n"], dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i1.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: i1.IonButtons, selector: "ion-buttons", inputs: ["collapse"] }, { kind: "component", type: i1.IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: i1.IonItem, selector: "ion-item", inputs: ["button", "color", "counter", "counterFormatter", "detail", "detailIcon", "disabled", "download", "fill", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "shape", "target", "type"] }, { kind: "component", type: i1.IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TapValueEditorContainerComponent, decorators: [{ type: Component, args: [{ selector: 'tap-value-editor-container', template: "<ion-item\n style=\"pointer-events: auto\"\n [lines]=\"lines\"\n [button]=\"button\"\n [ngClass]=\"{ 'item-input': editModeChange.value }\"\n>\n <ng-content></ng-content>\n <ion-buttons\n slot=\"end\"\n *ngIf=\"\n showEditButton ||\n showRefreshButton ||\n error ||\n pendingSubmitValue !== undefined\n \"\n >\n <ion-button\n (click)=\"explainWaitForSubmit(); $event.stopPropagation()\"\n *ngIf=\"pendingSubmitValue !== undefined\"\n [disabled]=\"false\"\n color=\"warning\"\n >\n <ion-icon name=\"alert-circle\"></ion-icon>\n </ion-button>\n <ion-button\n (click)=\"enableEditMode(); $event.stopPropagation()\"\n class=\"btn-enable-edit-mode\"\n *ngIf=\"\n pendingSubmitValue === undefined &&\n showEditButton &&\n !editModeChange.value\n \"\n >\n <ion-icon name=\"create\"></ion-icon>\n </ion-button>\n <ion-button\n (click)=\"submitValue(); $event.stopPropagation()\"\n *ngIf=\"\n showSubmitButton &&\n editModeChange.value &&\n pendingSubmitValue === undefined\n \"\n [disabled]=\"loadingChange | async\"\n class=\"btn-submit-value\"\n >\n <ion-icon name=\"send\"></ion-icon>\n </ion-button>\n <ion-button\n (click)=\"cancelEdit(); $event.stopPropagation()\"\n *ngIf=\"editModeChange.value || pendingSubmitValue !== undefined\"\n [disabled]=\"loadingChange | async\"\n class=\"btn-close-edit-mode\"\n >\n <ion-icon name=\"close\"></ion-icon>\n </ion-button>\n <ion-button\n *ngIf=\"showRefreshButton\"\n (click)=\"refreshValue(); $event.stopPropagation()\"\n [disabled]=\"loadingChange | async\"\n class=\"btn-refresh-value\"\n >\n <ion-icon name=\"sync\"></ion-icon>\n </ion-button>\n <ion-button\n *ngIf=\"error\"\n (click)=\"openErrorDialog(error); $event.stopPropagation()\"\n class=\"btn-show-error\"\n >\n <ion-icon slot=\"end\" name=\"alert-circle\" color=\"danger\"></ion-icon>\n </ion-button>\n </ion-buttons>\n <ion-spinner *ngIf=\"loadingChange | async\" slot=\"end\"></ion-spinner>\n</ion-item>\n", styles: ["ion-buttons[slot=end]{margin-inline-start:6px}ion-item{--ion-item-background: transparent;--color: initial}\n"] }] }], ctorParameters: function () { return [{ type: i1.AlertController }, { type: i2.CurrentDeviceService }, { type: i0.ChangeDetectorRef }, { type: i0.NgZone }, { type: i1.ModalController }]; }, propDecorators: { button: [{ type: Input }], variable: [{ type: Input }], inputOptions: [{ type: Input }], showRefreshButton: [{ type: Input }], showSubmitButton: [{ type: Input }], showEditButton: [{ type: Input }], error: [{ type: Input }], lines: [{ type: Input }], modalEdition: [{ type: Input }], value: [{ type: Input }], submit: [{ type: Output }], refresh: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tap-value-editor-container.component.js","sourceRoot":"","sources":["../../../../../../projects/iotize-ionic/src/lib/tap-value-editor-container/tap-value-editor-container.component.ts","../../../../../../projects/iotize-ionic/src/lib/tap-value-editor-container/tap-value-editor-container.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Output,\n} from '@angular/core';\nimport { AlertController, ModalController } from '@ionic/angular';\nimport { isCodeError } from '@iotize/common/error';\nimport { TapClientError } from '@iotize/tap/client/impl';\nimport { EditableValueDataStreamInterface } from '@iotize/tap/data';\nimport { BehaviorSubject, merge, Observable, of, Subject } from 'rxjs';\nimport { filter, shareReplay, startWith, switchMap, tap } from 'rxjs/operators';\nimport { CurrentDeviceService } from '../current-device.service';\nimport { LibError } from '../errors';\nimport { InlineEditorComponent } from '../inline-editor/inline-editor.component';\nimport { isNfcTapRequiredError } from '../nfc/utility';\nimport { PendingCallManager } from '../pending-call-manager';\nimport { TapValueEditorModalComponent } from '../tap-value-editor-modal/tap-value-editor-modal.component';\n\nfunction isTapNotConnectedError(err: Error) {\n  return (\n    isNfcTapRequiredError(err) ||\n    isCodeError(TapClientError.Code.NotConnectedError, err)\n  );\n}\n\n@Component({\n  selector: 'tap-value-editor-container',\n  templateUrl: './tap-value-editor-container.component.html',\n  styleUrls: ['./tap-value-editor-container.component.scss'],\n})\nexport class TapValueEditorContainerComponent implements OnInit, OnDestroy {\n  @Input() button: boolean = false;\n\n  @Input() set variable(\n    v: EditableValueDataStreamInterface<unknown, any> | undefined\n  ) {\n    this.variableChange.next(v);\n  }\n\n  @Input() inputOptions: InlineEditorComponent.InputOptions | any = {\n    type: 'text',\n  };\n\n  /**\n   * Show refresh button\n   */\n  @Input() showRefreshButton = false;\n\n  /**\n   * Show submit button\n   */\n  @Input() showSubmitButton = false;\n\n  /**\n   * Show edit button\n   */\n  @Input() showEditButton = false;\n\n  /**\n   * Enable edition mode\n   */\n  editModeChange = new BehaviorSubject<boolean>(false);\n\n  loadingChange = new BehaviorSubject<boolean>(false);\n\n  /**\n   * Error to display\n   */\n  @Input() error?: Error;\n\n  @Input() lines = 'none';\n\n  @Input() modalEdition: boolean = false;\n\n  /**\n   * Force value to display\n   */\n  @Input() set value(v: any) {\n    this.forceValueChange.next(v);\n  }\n\n  @Output() submit = new EventEmitter<any>();\n\n  @Output() refresh = new EventEmitter<any>();\n\n  pendingSubmitValue?: any;\n\n  valueToSubmit?: any;\n\n  lastReadValue?: any;\n\n  private variableChange = new BehaviorSubject<\n    EditableValueDataStreamInterface<unknown, any> | undefined\n  >(undefined);\n\n  public variableValuesStream: Observable<any> = this.variableChange.pipe(\n    filter((v) => !!v),\n    switchMap((v) => v!.values),\n    tap((v) => {\n      this.lastReadValue = v;\n    })\n  );\n\n  forceValueChange = new Subject<any>();\n\n  valueToDisplay = this.editModeChange.pipe(\n    switchMap((editMode) => {\n      if (editMode || this.pendingSubmitValue !== undefined) {\n        return of(this.valueToSubmit);\n      } else {\n        const lastValue = this.variableChange.value?.valueSnapshot;\n        return merge(\n          this.variableValuesStream.pipe(startWith(lastValue)),\n          this.forceValueChange\n        );\n      }\n    }),\n    shareReplay(1)\n  );\n\n  private pendingCallManager?: PendingCallManager;\n\n  private destroyed = new Subject<void>();\n\n  constructor(\n    private alertDialog: AlertController,\n    private tapService: CurrentDeviceService,\n    private changeDetectorRef: ChangeDetectorRef,\n    private ngZone: NgZone,\n    private modalController: ModalController\n  ) {}\n\n  ngOnInit() {\n    this.pendingCallManager = PendingCallManager.create(this.tapService);\n\n    this.pendingCallManager.pendingCallResult.subscribe((error) => {\n      this.pendingSubmitValue = undefined;\n      this.valueToSubmit = undefined;\n      this.editModeChange.next(false);\n      this.error = error;\n      this.changeDetectorRef.detectChanges();\n    });\n  }\n\n  // hasValueChanged(value: any) {\n  //   return this._lastValueFromDevice !== value;\n  // }\n\n  submitValue() {\n    if (this.valueToSubmit === undefined) {\n      console.warn('No value to submit yet');\n      return;\n    }\n    this._submitValue(this.valueToSubmit);\n  }\n\n  async notifyValueChange(newValue: any, forceWrite = false) {\n    if (forceWrite || this.lastReadValue !== newValue) {\n      await this._submitValue(newValue);\n    }\n  }\n\n  async openErrorDialog(error: Error) {\n    const dialog = await this.alertDialog.create({\n      header: 'Error',\n      message: error.message,\n      buttons: ['Ok'],\n    });\n    await dialog.present();\n  }\n\n  async explainWaitForSubmit() {\n    const alertDialog = await this.alertDialog.create({\n      header: 'Value not sent yet',\n      message: 'New value will be sent to the device once you are reconnected',\n      buttons: ['Ok'],\n    });\n    await alertDialog.present();\n  }\n\n  private async _submitValue(newValue: any) {\n    this.submit.emit(newValue);\n    // OLD System\n    const variable = this.variableChange.value;\n    if (!variable) {\n      this.error = LibError.componentArgumentRequired(\n        this.constructor.name,\n        'variable'\n      );\n      return;\n    }\n    if (!this.pendingCallManager) {\n      return;\n    }\n    console.log(`[${variable.id}]`, 'Submit value', newValue);\n    await this.startAction(\n      this.pendingCallManager\n        .exec(async () => {\n          const result = await variable.write(newValue);\n          this.editModeChange.next(false);\n          this.forceValueChange.next(newValue);\n          this.error = undefined;\n          this.pendingSubmitValue = undefined;\n          return result;\n        })\n        .catch((err: any) => {\n          if (!isTapNotConnectedError(err)) {\n            console.warn(\n              `[${variable.id}]`,\n              `onSubmit value \"${newValue}\" error`,\n              err\n            );\n            this.error = err;\n          } else {\n            this.pendingSubmitValue = newValue;\n            this.valueToSubmit = newValue;\n            this.editModeChange.next(false);\n          }\n        })\n    );\n  }\n\n  public async refreshValue() {\n    this.refresh.next(undefined);\n    const variable = this.variableChange.value;\n    if (variable) {\n      await this.startAction(\n        variable\n          .read()\n          .then((newValue: any) => {\n            this.forceValueChange.next(newValue);\n          })\n          .catch((err: any) => {\n            console.log(err);\n          })\n      );\n    }\n  }\n\n  async enableEditMode() {\n    this.modalEdition ? this.showEditModal() : this.editModeChange.next(true);\n  }\n\n  private async showEditModal() {\n    const lastValue = this.variableChange.value?.valueSnapshot;\n\n    const modal = await this.modalController.create({\n      component: TapValueEditorModalComponent,\n      componentProps: {\n        inputOptions: this.inputOptions,\n        value: lastValue,\n      },\n    });\n    await modal.present();\n    const result = await modal.onDidDismiss();\n    if (result) {\n      if (result.role === 'submit') {\n        const newValue = result.data.value;\n        await this.notifyValueChange(newValue);\n      }\n    }\n  }\n\n  private async startAction(actions: Promise<any>): Promise<any> {\n    this.ngZone.run(async () => {\n      this.loadingChange.next(true);\n      try {\n        return await actions;\n      } catch (err: any) {\n        console.warn('action failed', err);\n      } finally {\n        this.loadingChange.next(false);\n      }\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.destroyed.next();\n    this.destroyed.complete();\n    this.pendingCallManager?.destroy();\n  }\n\n  cancelEdit() {\n    this.pendingCallManager?.cancel();\n    this.error = undefined;\n    this.pendingSubmitValue = undefined;\n    this.valueToSubmit = undefined;\n    this.editModeChange.next(false);\n  }\n}\n","<ion-item\n  style=\"pointer-events: auto\"\n  [lines]=\"lines\"\n  [button]=\"button\"\n  [ngClass]=\"{ 'item-input': editModeChange.value }\"\n>\n  <ng-content></ng-content>\n  <ion-buttons\n    slot=\"end\"\n    *ngIf=\"\n      showEditButton ||\n      showRefreshButton ||\n      error ||\n      pendingSubmitValue !== undefined\n    \"\n  >\n    <ion-button\n      (click)=\"explainWaitForSubmit(); $event.stopPropagation()\"\n      *ngIf=\"pendingSubmitValue !== undefined\"\n      [disabled]=\"false\"\n      color=\"warning\"\n    >\n      <ion-icon name=\"alert-circle\"></ion-icon>\n    </ion-button>\n    <ion-button\n      (click)=\"enableEditMode(); $event.stopPropagation()\"\n      class=\"btn-enable-edit-mode\"\n      *ngIf=\"\n        pendingSubmitValue === undefined &&\n        showEditButton &&\n        !editModeChange.value\n      \"\n    >\n      <ion-icon name=\"create\"></ion-icon>\n    </ion-button>\n    <ion-button\n      (click)=\"submitValue(); $event.stopPropagation()\"\n      *ngIf=\"\n        showSubmitButton &&\n        editModeChange.value &&\n        pendingSubmitValue === undefined\n      \"\n      [disabled]=\"loadingChange | async\"\n      class=\"btn-submit-value\"\n    >\n      <ion-icon name=\"send\"></ion-icon>\n    </ion-button>\n    <ion-button\n      (click)=\"cancelEdit(); $event.stopPropagation()\"\n      *ngIf=\"editModeChange.value || pendingSubmitValue !== undefined\"\n      [disabled]=\"loadingChange | async\"\n      class=\"btn-close-edit-mode\"\n    >\n      <ion-icon name=\"close\"></ion-icon>\n    </ion-button>\n    <ion-button\n      *ngIf=\"showRefreshButton\"\n      (click)=\"refreshValue(); $event.stopPropagation()\"\n      [disabled]=\"loadingChange | async\"\n      class=\"btn-refresh-value\"\n    >\n      <ion-icon name=\"sync\"></ion-icon>\n    </ion-button>\n    <ion-button\n      *ngIf=\"error\"\n      (click)=\"openErrorDialog(error); $event.stopPropagation()\"\n      class=\"btn-show-error\"\n    >\n      <ion-icon slot=\"end\" name=\"alert-circle\" color=\"danger\"></ion-icon>\n    </ion-button>\n  </ion-buttons>\n  <ion-spinner *ngIf=\"loadingChange | async\" slot=\"end\"></ion-spinner>\n</ion-item>\n"]}