@iotize/ionic
Version:
Iotize specific building blocks on top of @ionic/angular.
314 lines • 49.6 kB
JavaScript
import { __decorate, __metadata } from "tslib";
import { Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
import { ModalController } from '@ionic/angular';
import { deepCopy } from '@iotize/common/utility';
import { TapValueEditorContainerComponent } from '@iotize/ionic';
import { NumberConverter, TapStreamReader, TapStreamWriter, } from '@iotize/tap/client/impl';
import { variableDataTypeToByteSize, } from '@iotize/tap/data';
import { VariableType } from '@iotize/tap/service/impl/variable';
import { BehaviorSubject, NEVER, Subject } from 'rxjs';
import { switchMap, takeUntil } from 'rxjs/operators';
import { MonitoringAppGenComponent } from '../../metadata/decorators';
import { TapVariableBitsEditorInfoModalComponent } from './tap-variable-bits-editor-info-modal/tap-variable-bits-editor-info-modal.component';
import * as i0 from "@angular/core";
import * as i1 from "@ionic/angular";
import * as i2 from "@angular/common";
import * as i3 from "@iotize/ionic";
let TapVariableBitsEditorComponent = class TapVariableBitsEditorComponent {
set variable(variable) {
this._variableChange.next(variable);
}
set bitsTemplate(template) {
template = sanitizeBitsTemplateParameter(template);
this._bitsTemplate = template;
if (template) {
this.bitsArrayEditable = deepCopy(template);
}
else {
this.bitsArrayEditable = [];
}
this._restoreLastValueReceived();
}
set value(v) {
if (v === undefined) {
v = 0;
}
this._setRawValue(v);
}
constructor(modalController) {
this.modalController = modalController;
this.bitsArrayEditable = [];
this._variableChange = new BehaviorSubject(undefined);
this.dataSuccessfullyUpdated = new EventEmitter();
this.showHelpButton = true;
this.bitOffColor = 'danger';
this.bitOnColor = 'success';
this.editable = true;
this.editButtonsColor = 'primary';
this.numberByteLength = 4;
this.destroyed = new Subject();
this.loaded = false;
this.onSaving = false;
this.editMode = false;
}
ngOnDestroy() {
this.destroyed.next();
}
ngOnInit() {
this._variableChange
.pipe(switchMap((variable) => {
if (variable) {
return variable.values;
}
else {
return NEVER;
}
}), takeUntil(this.destroyed))
.subscribe((rawVariableValue) => {
this._setRawValue(rawVariableValue);
});
}
_setRawValue(value) {
const rawValue = this._valueToBuffer(value);
this.rawStored = rawValue;
if (!this.editMode) {
this.rawToView(rawValue);
this.loaded = true;
}
}
rawToView(rawVariableValue) {
const tapStreamReader = TapStreamReader.fromArray(rawVariableValue);
if (this.bitsArrayEditable.length === 0) {
this._initBitsTemplate(rawVariableValue);
}
const bitSize = rawVariableValue.length * 8 - 1;
this.bitsArrayEditable.forEach((row) => {
for (let bit of row) {
const bitPosition = this._bitIndexToBufferPosition(bitSize, bit.index);
tapStreamReader.setBitPosition(bitPosition);
bit.value = tapStreamReader.readBoolean(1);
}
});
}
_initBitsTemplate(raw) {
let template = this._bitsTemplate;
if (!Array.isArray(template)) {
template = [[]];
for (let i = 0; i < raw.length * 8; i++) {
template[0].push({
description: 'No description',
label: i.toString(),
index: i,
});
}
}
for (let i = 0; i < template.length; i++) {
this.bitsArrayEditable.push([]);
for (let bit of template[i]) {
this.bitsArrayEditable[i].push({
...bit,
});
}
}
}
cancelEdit() {
this.editMode = false;
this._restoreLastValueReceived();
}
onEditMode() {
this.editMode = true;
}
async onClickEdit(bit) {
bit.value = !bit.value;
}
async onSaveEdit(variable) {
try {
if (!variable) {
return;
}
const variableByteLength = this.getNumberByteLength();
const bufferValue = this._userInputToBuffer(variableByteLength);
this.editMode = false;
this.onSaving = true;
if (typeof variable.writeRaw === 'function') {
await variable.writeRaw(bufferValue);
}
else {
const decodedValue = variable.converter
? variable.converter.decode(bufferValue)
: bufferValue;
await variable.write(decodedValue);
}
this.dataSuccessfullyUpdated.next('Data successfully updated !');
}
catch (error) {
this._restoreLastValueReceived();
this.ctx.error = error;
}
this.onSaving = false;
}
getNumberByteLength() {
return this.numberByteLength
? this.numberByteLength
: this._guessDataLength();
}
_restoreLastValueReceived() {
if (this.rawStored) {
this.rawToView(this.rawStored);
}
}
_valueToBuffer(value) {
if (typeof value === 'number') {
const variableByteLength = this.getNumberByteLength();
if (variableByteLength > 4) {
throw new Error(`Variable on ${variableByteLength} bytes are not supported.`);
}
else if (variableByteLength === 0) {
return new Uint8Array([]);
}
else {
const converter = new NumberConverter({
sizeOf: (variableByteLength * 8),
signed: false,
leastSignificantBitFirst: false,
});
return converter.encode(value);
}
}
else {
return value;
}
}
_bitIndexToBufferPosition(bitLength, index) {
return this.msb ? index : bitLength - index;
}
_userInputToBuffer(byteLength) {
const tapStreamWriter = TapStreamWriter.create(byteLength);
const bitLength = byteLength * 8 - 1;
this.bitsArrayEditable.forEach((row) => {
for (let bit of row) {
tapStreamWriter.setBitPosition(this._bitIndexToBufferPosition(bitLength, bit.index));
tapStreamWriter.writeBoolean(bit.value || false, 1);
}
});
tapStreamWriter.setBitPosition(bitLength);
const result = tapStreamWriter.toBytes;
console.log('DEBUG USER INPUT TO BUFFER', bitLength, deepCopy(this.bitsArrayEditable), 'result', result);
if (result.length !== byteLength) {
console.warn(`TapVariableBitsEditorComponent encoded value to write does not have the expected byte length of ${byteLength} bytes:`, result);
}
return result;
}
/**
* @deprecated use numberByteLength parameter instead
* @param variable
* @returns
*/
_guessDataLength(variable = this._variableChange.value) {
if (!variable) {
console.warn(`Cannot guess data length, variable is not set yet`);
return 0;
}
const tapVariableConfig = variable
.config;
const quantity = tapVariableConfig?.length || 1;
let unit;
if (tapVariableConfig?.dataType === undefined) {
// TODO fix with API change
console.warn(`TapVariableBitsEditorComponent failed to guess variable size. use 1 byte size. It may trigger error if size is not valid.`, variable);
unit = VariableType.Data.UINT32;
}
else {
unit = tapVariableConfig.dataType;
}
const length = quantity * variableDataTypeToByteSize(unit);
return length;
}
async showInfo() {
this._infoModal = this.modalController.create({
component: TapVariableBitsEditorInfoModalComponent,
componentProps: {
bitOnColor: this.bitOnColor,
bitOffColor: this.bitOffColor,
rows: this.bitsArrayEditable,
},
});
try {
const modal = await await this._infoModal;
modal.present();
await modal.onDidDismiss();
}
finally {
this._infoModal = undefined;
}
}
};
/** @nocollapse */ TapVariableBitsEditorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TapVariableBitsEditorComponent, deps: [{ token: i1.ModalController }], target: i0.ɵɵFactoryTarget.Component });
/** @nocollapse */ TapVariableBitsEditorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: TapVariableBitsEditorComponent, selector: "tap-variable-bits-editor", inputs: { variable: "variable", showHelpButton: "showHelpButton", bitOffColor: "bitOffColor", bitOnColor: "bitOnColor", editable: "editable", editButtonsColor: "editButtonsColor", msb: "msb", numberByteLength: "numberByteLength", bitsTemplate: "bitsTemplate", value: "value" }, outputs: { dataSuccessfullyUpdated: "dataSuccessfullyUpdated" }, viewQueries: [{ propertyName: "ctx", first: true, predicate: ["ctx"], descendants: true }], ngImport: i0, template: "<tap-value-editor-container\n #ctx\n [variable]=\"_variableChange.value\"\n [showEditButton]=\"false\"\n [showSubmitButton]=\"false\"\n [lines]=\"'none'\"\n *ngIf=\"bitsArrayEditable.length > 0; else noBitsTemplate\"\n>\n <ion-grid id=\"ionGrid\">\n <ion-row *ngFor=\"let row of bitsArrayEditable\" class=\"row-bits\">\n <div class=\"bit-div\" *ngFor=\"let bit of row\">\n <ion-fab-button\n (click)=\"onClickEdit(bit)\"\n class=\"fab-row\"\n size=\"small\"\n [color]=\"\n (bit.value && !bit.reverse) || (!bit.value && bit.reverse)\n ? bitOnColor\n : bitOffColor\n \"\n [disabled]=\"!editMode\"\n >\n </ion-fab-button>\n <ion-label class=\"label-under\">\n {{ bit.label }}\n </ion-label>\n </div>\n </ion-row>\n <ion-item lines=\"none\" *ngIf=\"editable || showHelpButton\">\n <ion-buttons id=\"buttonsBottom\">\n <ion-button\n fill=\"clear\"\n (click)=\"showInfo()\"\n [color]=\"editButtonsColor\"\n *ngIf=\"showHelpButton\"\n >\n <ion-icon\n class=\"button-edit-icon\"\n name=\"information-circle\"\n ></ion-icon>\n </ion-button>\n <ng-container *ngIf=\"editable\">\n <ion-button\n [color]=\"editButtonsColor\"\n fill=\"clear\"\n [disabled]=\"onSaving\"\n (click)=\"onEditMode()\"\n *ngIf=\"!editMode\"\n >\n <ion-icon class=\"button-edit-icon\" name=\"pencil\"></ion-icon>\n </ion-button>\n <ion-button\n fill=\"clear\"\n (click)=\"cancelEdit()\"\n *ngIf=\"editMode\"\n [color]=\"editButtonsColor\"\n >\n <ion-icon class=\"button-edit-icon\" name=\"close\"></ion-icon>\n </ion-button>\n <ion-button\n fill=\"clear\"\n [color]=\"editButtonsColor\"\n [disabled]=\"!editMode\"\n (click)=\"onSaveEdit(_variableChange.value)\"\n *ngIf=\"!onSaving; else saveLoader\"\n >\n <ion-icon class=\"button-edit-icon\" name=\"save\"></ion-icon>\n </ion-button>\n <ng-template #saveLoader>\n <ion-spinner\n [color]=\"editButtonsColor\"\n name=\"crescent\"\n ></ion-spinner> </ng-template\n ></ng-container>\n </ion-buttons>\n </ion-item>\n </ion-grid>\n</tap-value-editor-container>\n<ng-template #noBitsTemplate>\n <ion-progress-bar type=\"indeterminate\"></ion-progress-bar>\n</ng-template>\n", styles: [".checkbox-bit{--border-radius: 5px }ion-label{opacity:1!important}ion-checkbox{opacity:1!important}.fab-row{height:2em;width:2em;margin:.3vw}.row-bits{display:flex;justify-content:center}.icon-bit{font-size:2.3em}ion-fab-button{opacity:1!important}.button-edit-icon{font-size:2em}.span-description{white-space:pre-line!important}.bit-label-info{font-weight:500}.bit-div{text-align:center;margin:2px}.label-under{writing-mode:vertical-lr;transform:rotate(180deg);margin:auto;font-family:system-ui;padding-left:4px}#buttonsBottom{margin:auto}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.TapValueEditorContainerComponent, selector: "tap-value-editor-container", inputs: ["button", "variable", "inputOptions", "showRefreshButton", "showSubmitButton", "showEditButton", "error", "lines", "modalEdition", "value"], outputs: ["submit", "refresh"] }, { 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.IonFabButton, selector: "ion-fab-button", inputs: ["activated", "closeIcon", "color", "disabled", "download", "href", "mode", "rel", "routerAnimation", "routerDirection", "show", "size", "target", "translucent", "type"] }, { kind: "component", type: i1.IonGrid, selector: "ion-grid", inputs: ["fixed"] }, { 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.IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: i1.IonProgressBar, selector: "ion-progress-bar", inputs: ["buffer", "color", "mode", "reversed", "type", "value"] }, { kind: "component", type: i1.IonRow, selector: "ion-row" }, { kind: "component", type: i1.IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }] });
TapVariableBitsEditorComponent = __decorate([
MonitoringAppGenComponent({
constraints: {
isNumber: 'YES',
isArray: 'NO',
},
events: [
{
name: 'submit',
dataType: 'number',
},
],
}),
__metadata("design:paramtypes", [ModalController])
], TapVariableBitsEditorComponent);
export { TapVariableBitsEditorComponent };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: TapVariableBitsEditorComponent, decorators: [{
type: Component,
args: [{ selector: 'tap-variable-bits-editor', template: "<tap-value-editor-container\n #ctx\n [variable]=\"_variableChange.value\"\n [showEditButton]=\"false\"\n [showSubmitButton]=\"false\"\n [lines]=\"'none'\"\n *ngIf=\"bitsArrayEditable.length > 0; else noBitsTemplate\"\n>\n <ion-grid id=\"ionGrid\">\n <ion-row *ngFor=\"let row of bitsArrayEditable\" class=\"row-bits\">\n <div class=\"bit-div\" *ngFor=\"let bit of row\">\n <ion-fab-button\n (click)=\"onClickEdit(bit)\"\n class=\"fab-row\"\n size=\"small\"\n [color]=\"\n (bit.value && !bit.reverse) || (!bit.value && bit.reverse)\n ? bitOnColor\n : bitOffColor\n \"\n [disabled]=\"!editMode\"\n >\n </ion-fab-button>\n <ion-label class=\"label-under\">\n {{ bit.label }}\n </ion-label>\n </div>\n </ion-row>\n <ion-item lines=\"none\" *ngIf=\"editable || showHelpButton\">\n <ion-buttons id=\"buttonsBottom\">\n <ion-button\n fill=\"clear\"\n (click)=\"showInfo()\"\n [color]=\"editButtonsColor\"\n *ngIf=\"showHelpButton\"\n >\n <ion-icon\n class=\"button-edit-icon\"\n name=\"information-circle\"\n ></ion-icon>\n </ion-button>\n <ng-container *ngIf=\"editable\">\n <ion-button\n [color]=\"editButtonsColor\"\n fill=\"clear\"\n [disabled]=\"onSaving\"\n (click)=\"onEditMode()\"\n *ngIf=\"!editMode\"\n >\n <ion-icon class=\"button-edit-icon\" name=\"pencil\"></ion-icon>\n </ion-button>\n <ion-button\n fill=\"clear\"\n (click)=\"cancelEdit()\"\n *ngIf=\"editMode\"\n [color]=\"editButtonsColor\"\n >\n <ion-icon class=\"button-edit-icon\" name=\"close\"></ion-icon>\n </ion-button>\n <ion-button\n fill=\"clear\"\n [color]=\"editButtonsColor\"\n [disabled]=\"!editMode\"\n (click)=\"onSaveEdit(_variableChange.value)\"\n *ngIf=\"!onSaving; else saveLoader\"\n >\n <ion-icon class=\"button-edit-icon\" name=\"save\"></ion-icon>\n </ion-button>\n <ng-template #saveLoader>\n <ion-spinner\n [color]=\"editButtonsColor\"\n name=\"crescent\"\n ></ion-spinner> </ng-template\n ></ng-container>\n </ion-buttons>\n </ion-item>\n </ion-grid>\n</tap-value-editor-container>\n<ng-template #noBitsTemplate>\n <ion-progress-bar type=\"indeterminate\"></ion-progress-bar>\n</ng-template>\n", styles: [".checkbox-bit{--border-radius: 5px }ion-label{opacity:1!important}ion-checkbox{opacity:1!important}.fab-row{height:2em;width:2em;margin:.3vw}.row-bits{display:flex;justify-content:center}.icon-bit{font-size:2.3em}ion-fab-button{opacity:1!important}.button-edit-icon{font-size:2em}.span-description{white-space:pre-line!important}.bit-label-info{font-weight:500}.bit-div{text-align:center;margin:2px}.label-under{writing-mode:vertical-lr;transform:rotate(180deg);margin:auto;font-family:system-ui;padding-left:4px}#buttonsBottom{margin:auto}\n"] }]
}], ctorParameters: function () { return [{ type: i1.ModalController }]; }, propDecorators: { ctx: [{
type: ViewChild,
args: ['ctx']
}], dataSuccessfullyUpdated: [{
type: Output
}], variable: [{
type: Input
}], showHelpButton: [{
type: Input
}], bitOffColor: [{
type: Input
}], bitOnColor: [{
type: Input
}], editable: [{
type: Input
}], editButtonsColor: [{
type: Input
}], msb: [{
type: Input
}], numberByteLength: [{
type: Input
}], bitsTemplate: [{
type: Input
}], value: [{
type: Input
}] } });
export function sanitizeBitsTemplateParameter(template) {
if (Array.isArray(template) &&
template.length > 0 &&
!Array.isArray(template[0])) {
template = [template];
}
if (template && !Array.isArray(template)) {
template = [];
}
if (template) {
template = template.filter((row) => {
return !!row && Array.isArray(row);
});
template.forEach((row, index) => {
template[index] = row.filter((item) => {
return !!item && typeof item === 'object';
});
});
}
return template;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tap-variable-bits-editor.component.js","sourceRoot":"","sources":["../../../../../../../../projects/iotize-ionic/monitoring/src/lib/ui-components/tap-variable-bits-editor/tap-variable-bits-editor.component.ts","../../../../../../../../projects/iotize-ionic/monitoring/src/lib/ui-components/tap-variable-bits-editor/tap-variable-bits-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  ViewChild,\n} from '@angular/core';\nimport { ModalController } from '@ionic/angular';\nimport { deepCopy } from '@iotize/common/utility';\nimport { TapValueEditorContainerComponent } from '@iotize/ionic';\nimport {\n  NumberConverter,\n  TapStreamReader,\n  TapStreamWriter,\n} from '@iotize/tap/client/impl';\nimport {\n  AbstractVariable,\n  TapVariable,\n  variableDataTypeToByteSize,\n} from '@iotize/tap/data';\nimport { KeyTypeType } from '@iotize/tap/data/lib/utility/editable-data-stream';\nimport { VariableType } from '@iotize/tap/service/impl/variable';\nimport { BehaviorSubject, NEVER, Subject } from 'rxjs';\nimport { switchMap, takeUntil } from 'rxjs/operators';\nimport { MonitoringAppGenComponent } from '../../metadata/decorators';\nimport { Bit, BitTemplate } from './definitions';\nimport { TapVariableBitsEditorInfoModalComponent } from './tap-variable-bits-editor-info-modal/tap-variable-bits-editor-info-modal.component';\n\n@MonitoringAppGenComponent({\n  constraints: {\n    isNumber: 'YES',\n    isArray: 'NO',\n  },\n  events: [\n    {\n      name: 'submit',\n      dataType: 'number',\n    },\n  ],\n})\n@Component({\n  selector: 'tap-variable-bits-editor',\n  templateUrl: './tap-variable-bits-editor.component.html',\n  styleUrls: ['./tap-variable-bits-editor.component.scss'],\n})\nexport class TapVariableBitsEditorComponent implements OnInit, OnDestroy {\n  @ViewChild('ctx') ctx!: TapValueEditorContainerComponent;\n\n  bitsArrayEditable: Bit[][] = [];\n\n  _variableChange = new BehaviorSubject<\n    AbstractVariable<number | Uint8Array> | undefined\n  >(undefined);\n\n  @Output() dataSuccessfullyUpdated = new EventEmitter<string>();\n\n  @Input()\n  set variable(variable: AbstractVariable<number | Uint8Array> | undefined) {\n    this._variableChange.next(variable);\n  }\n\n  @Input() showHelpButton = true;\n\n  @Input()\n  bitOffColor: string = 'danger';\n\n  @Input()\n  bitOnColor: string = 'success';\n\n  @Input()\n  editable = true;\n\n  @Input()\n  editButtonsColor?: string = 'primary';\n\n  @Input() msb?: boolean;\n\n  @Input() numberByteLength: 1 | 2 | 4 = 4;\n\n  @Input()\n  set bitsTemplate(template: BitTemplate[][] | undefined) {\n    template = sanitizeBitsTemplateParameter(template);\n    this._bitsTemplate = template;\n    if (template) {\n      this.bitsArrayEditable = deepCopy(template);\n    } else {\n      this.bitsArrayEditable = [];\n    }\n    this._restoreLastValueReceived();\n  }\n\n  @Input()\n  set value(v: number | Uint8Array | undefined) {\n    if (v === undefined) {\n      v = 0;\n    }\n    this._setRawValue(v);\n  }\n\n  private _bitsTemplate: BitTemplate[][] | undefined;\n\n  private destroyed = new Subject<void>();\n\n  loaded: boolean = false;\n  onSaving: boolean = false;\n  editMode: boolean = false;\n\n  rawStored?: Uint8Array;\n\n  private _infoModal?: Promise<HTMLIonModalElement>;\n\n  constructor(private modalController: ModalController) {}\n\n  ngOnDestroy(): void {\n    this.destroyed.next();\n  }\n\n  ngOnInit() {\n    this._variableChange\n      .pipe(\n        switchMap((variable) => {\n          if (variable) {\n            return variable.values;\n          } else {\n            return NEVER;\n          }\n        }),\n        takeUntil(this.destroyed)\n      )\n      .subscribe((rawVariableValue) => {\n        this._setRawValue(rawVariableValue);\n      });\n  }\n\n  private _setRawValue(value: Uint8Array | number) {\n    const rawValue = this._valueToBuffer(value);\n    this.rawStored = rawValue;\n    if (!this.editMode) {\n      this.rawToView(rawValue);\n      this.loaded = true;\n    }\n  }\n\n  private rawToView(rawVariableValue: Uint8Array) {\n    const tapStreamReader = TapStreamReader.fromArray(rawVariableValue);\n    if (this.bitsArrayEditable.length === 0) {\n      this._initBitsTemplate(rawVariableValue);\n    }\n    const bitSize = rawVariableValue.length * 8 - 1;\n    this.bitsArrayEditable.forEach((row) => {\n      for (let bit of row) {\n        const bitPosition = this._bitIndexToBufferPosition(bitSize, bit.index);\n        tapStreamReader.setBitPosition(bitPosition);\n        bit.value = tapStreamReader.readBoolean(1);\n      }\n    });\n  }\n\n  private _initBitsTemplate(raw: Uint8Array) {\n    let template = this._bitsTemplate;\n    if (!Array.isArray(template)) {\n      template = [[]];\n      for (let i = 0; i < raw.length * 8; i++) {\n        template[0].push({\n          description: 'No description',\n          label: i.toString(),\n          index: i,\n        });\n      }\n    }\n    for (let i = 0; i < template.length; i++) {\n      this.bitsArrayEditable.push([]);\n      for (let bit of template[i]) {\n        this.bitsArrayEditable[i].push({\n          ...bit,\n        });\n      }\n    }\n  }\n\n  cancelEdit() {\n    this.editMode = false;\n    this._restoreLastValueReceived();\n  }\n\n  onEditMode() {\n    this.editMode = true;\n  }\n\n  async onClickEdit(bit: Bit) {\n    bit.value = !bit.value;\n  }\n\n  async onSaveEdit(\n    variable: AbstractVariable<number | Uint8Array> | undefined\n  ) {\n    try {\n      if (!variable) {\n        return;\n      }\n      const variableByteLength = this.getNumberByteLength();\n      const bufferValue = this._userInputToBuffer(variableByteLength);\n      this.editMode = false;\n      this.onSaving = true;\n      if (typeof variable.writeRaw === 'function') {\n        await variable.writeRaw(bufferValue);\n      } else {\n        const decodedValue = variable.converter\n          ? variable.converter.decode(bufferValue)\n          : bufferValue;\n        await variable.write(decodedValue);\n      }\n      this.dataSuccessfullyUpdated.next('Data successfully updated !');\n    } catch (error) {\n      this._restoreLastValueReceived();\n      this.ctx.error = error as Error;\n    }\n    this.onSaving = false;\n  }\n\n  private getNumberByteLength() {\n    return this.numberByteLength\n      ? this.numberByteLength\n      : this._guessDataLength();\n  }\n\n  private _restoreLastValueReceived() {\n    if (this.rawStored) {\n      this.rawToView(this.rawStored);\n    }\n  }\n\n  private _valueToBuffer(value: number | Uint8Array) {\n    if (typeof value === 'number') {\n      const variableByteLength = this.getNumberByteLength();\n      if (variableByteLength > 4) {\n        throw new Error(\n          `Variable on ${variableByteLength} bytes are not supported.`\n        );\n      } else if (variableByteLength === 0) {\n        return new Uint8Array([]);\n      } else {\n        const converter = new NumberConverter({\n          sizeOf: (variableByteLength * 8) as 8 | 16 | 32,\n          signed: false,\n          leastSignificantBitFirst: false,\n        });\n        return converter.encode(value);\n      }\n    } else {\n      return value;\n    }\n  }\n\n  private _bitIndexToBufferPosition(bitLength: number, index: number) {\n    return this.msb ? index : bitLength - index;\n  }\n\n  private _userInputToBuffer(byteLength: number): Uint8Array {\n    const tapStreamWriter = TapStreamWriter.create(byteLength);\n    const bitLength = byteLength * 8 - 1;\n    this.bitsArrayEditable.forEach((row) => {\n      for (let bit of row) {\n        tapStreamWriter.setBitPosition(\n          this._bitIndexToBufferPosition(bitLength, bit.index)\n        );\n        tapStreamWriter.writeBoolean(bit.value || false, 1);\n      }\n    });\n    tapStreamWriter.setBitPosition(bitLength);\n    const result = tapStreamWriter.toBytes;\n    console.log(\n      'DEBUG USER INPUT TO BUFFER',\n      bitLength,\n      deepCopy(this.bitsArrayEditable),\n      'result',\n      result\n    );\n    if (result.length !== byteLength) {\n      console.warn(\n        `TapVariableBitsEditorComponent encoded value to write does not have the expected byte length of ${byteLength} bytes:`,\n        result\n      );\n    }\n    return result;\n  }\n\n  /**\n   * @deprecated use numberByteLength parameter instead\n   * @param variable\n   * @returns\n   */\n  private _guessDataLength(variable = this._variableChange.value): number {\n    if (!variable) {\n      console.warn(`Cannot guess data length, variable is not set yet`);\n      return 0;\n    }\n    const tapVariableConfig = (variable as TapVariable<unknown, KeyTypeType>)\n      .config;\n    const quantity = tapVariableConfig?.length || 1;\n    let unit: VariableType.Data;\n    if (tapVariableConfig?.dataType === undefined) {\n      // TODO fix with API change\n      console.warn(\n        `TapVariableBitsEditorComponent failed to guess variable size. use 1 byte size. It may trigger error if size is not valid.`,\n        variable\n      );\n      unit = VariableType.Data.UINT32;\n    } else {\n      unit = tapVariableConfig.dataType;\n    }\n    const length = quantity * variableDataTypeToByteSize(unit);\n    return length;\n  }\n\n  async showInfo() {\n    this._infoModal = this.modalController.create({\n      component: TapVariableBitsEditorInfoModalComponent,\n      componentProps: {\n        bitOnColor: this.bitOnColor,\n        bitOffColor: this.bitOffColor,\n        rows: this.bitsArrayEditable,\n      },\n    });\n    try {\n      const modal = await await this._infoModal;\n      modal.present();\n      await modal.onDidDismiss();\n    } finally {\n      this._infoModal = undefined;\n    }\n  }\n}\n\nexport function sanitizeBitsTemplateParameter(\n  template: BitTemplate[][] | undefined\n): BitTemplate[][] | undefined {\n  if (\n    Array.isArray(template) &&\n    template.length > 0 &&\n    !Array.isArray(template[0])\n  ) {\n    template = [template as unknown as BitTemplate[]];\n  }\n  if (template && !Array.isArray(template)) {\n    template = [];\n  }\n  if (template) {\n    template = template.filter((row) => {\n      return !!row && Array.isArray(row);\n    });\n    template.forEach((row, index) => {\n      template!![index] = row.filter((item) => {\n        return !!item && typeof item === 'object';\n      });\n    });\n  }\n  return template;\n}\n","<tap-value-editor-container\n  #ctx\n  [variable]=\"_variableChange.value\"\n  [showEditButton]=\"false\"\n  [showSubmitButton]=\"false\"\n  [lines]=\"'none'\"\n  *ngIf=\"bitsArrayEditable.length > 0; else noBitsTemplate\"\n>\n  <ion-grid id=\"ionGrid\">\n    <ion-row *ngFor=\"let row of bitsArrayEditable\" class=\"row-bits\">\n      <div class=\"bit-div\" *ngFor=\"let bit of row\">\n        <ion-fab-button\n          (click)=\"onClickEdit(bit)\"\n          class=\"fab-row\"\n          size=\"small\"\n          [color]=\"\n            (bit.value && !bit.reverse) || (!bit.value && bit.reverse)\n              ? bitOnColor\n              : bitOffColor\n          \"\n          [disabled]=\"!editMode\"\n        >\n        </ion-fab-button>\n        <ion-label class=\"label-under\">\n          {{ bit.label }}\n        </ion-label>\n      </div>\n    </ion-row>\n    <ion-item lines=\"none\" *ngIf=\"editable || showHelpButton\">\n      <ion-buttons id=\"buttonsBottom\">\n        <ion-button\n          fill=\"clear\"\n          (click)=\"showInfo()\"\n          [color]=\"editButtonsColor\"\n          *ngIf=\"showHelpButton\"\n        >\n          <ion-icon\n            class=\"button-edit-icon\"\n            name=\"information-circle\"\n          ></ion-icon>\n        </ion-button>\n        <ng-container *ngIf=\"editable\">\n          <ion-button\n            [color]=\"editButtonsColor\"\n            fill=\"clear\"\n            [disabled]=\"onSaving\"\n            (click)=\"onEditMode()\"\n            *ngIf=\"!editMode\"\n          >\n            <ion-icon class=\"button-edit-icon\" name=\"pencil\"></ion-icon>\n          </ion-button>\n          <ion-button\n            fill=\"clear\"\n            (click)=\"cancelEdit()\"\n            *ngIf=\"editMode\"\n            [color]=\"editButtonsColor\"\n          >\n            <ion-icon class=\"button-edit-icon\" name=\"close\"></ion-icon>\n          </ion-button>\n          <ion-button\n            fill=\"clear\"\n            [color]=\"editButtonsColor\"\n            [disabled]=\"!editMode\"\n            (click)=\"onSaveEdit(_variableChange.value)\"\n            *ngIf=\"!onSaving; else saveLoader\"\n          >\n            <ion-icon class=\"button-edit-icon\" name=\"save\"></ion-icon>\n          </ion-button>\n          <ng-template #saveLoader>\n            <ion-spinner\n              [color]=\"editButtonsColor\"\n              name=\"crescent\"\n            ></ion-spinner> </ng-template\n        ></ng-container>\n      </ion-buttons>\n    </ion-item>\n  </ion-grid>\n</tap-value-editor-container>\n<ng-template #noBitsTemplate>\n  <ion-progress-bar type=\"indeterminate\"></ion-progress-bar>\n</ng-template>\n"]}