UNPKG

@iotize/ionic

Version:

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

187 lines 25.6 kB
import { ChangeDetectorRef, Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core'; import { FormBuilder, Validators, } from '@angular/forms'; import { IonInput } from '@ionic/angular'; import { Observable, Subject } from 'rxjs'; import { debug } from '../logger'; import { getFormFieldOrError } from '../utility'; import * as i0 from "@angular/core"; import * as i1 from "@angular/forms"; import * as i2 from "@angular/common"; import * as i3 from "@ionic/angular"; import * as i4 from "./value-or-default.pipe"; const TAG = 'InlineEditorComponent'; export class InlineEditorComponent { changeDetectorRef; /** * Emitter when form is submit */ onSubmit = new EventEmitter(); valueChange = new EventEmitter(); /** * Enable/disable edit mode */ set editMode(v) { this._editMode = v; this.field.setValue(this.currentValue); if (v) { this.enableEditMode(); } } editModeChange = new EventEmitter(); /** * True if field can be edited */ editable = false; unitValue; /** * Input option * You can defined how input will be rendered: text, number, toggle, select, ... */ inputOptions = { type: 'text', }; /** * Additional style for value */ valueStyle; /** * Set observable to subscribe to get value updates */ set value(obs) { this.valueSubscription?.unsubscribe(); if (obs && obs instanceof Observable) { this.valueSubscription = obs.subscribe((value) => { this.updateValue(value); }); } else if (obs !== undefined) { this.updateValue(obs); } } /** * Set value snapshot */ set forceValue(value) { this.updateValue(value); } /** * Enable/disable loader when submitting a value */ showLoader = true; /** * Set field error message */ set fieldError(err) { if (err) { debug(TAG, 'Set field error', err); this.field.setErrors({ submit: err.message, }); } else { this.field.setErrors(null); } } inputView; originalValue; form; _editMode = false; currentValue; valueSubscription; field; destroyed = new Subject(); constructor(formBuilder, changeDetectorRef) { this.changeDetectorRef = changeDetectorRef; this.form = formBuilder.group({ field: ['', Validators.required], }); this.field = getFormFieldOrError(this.form, 'field'); } ngOnDestroy() { this.destroyed.next(); this.destroyed.complete(); if (this.valueSubscription) { this.valueSubscription.unsubscribe(); } } onValueSubmit() { const value = this.field.value; if (this.field.errors) { return; } this.onSubmit.emit(value); } isEdited() { return this.field.value !== this.originalValue; } restoreValue() { this.setFieldValue(this.originalValue); } setFieldValue(value) { // let formattedValue = this.formatter() // this.inputFormatter ? this._info.editFormatter.write(value) : value; this.field.setValue(value || ''); } async onValueClick() { if (this.editable) { await this.enableEditMode(); this.editModeChange.next(true); } } onCancelEdit() { this._editMode = false; this.editModeChange.next(false); } async enableEditMode() { this._editMode = true; await this.setInputFocus(); } async setInputFocus() { this.changeDetectorRef.detectChanges(); await this.inputView.setFocus(); } updateValue(value) { this.currentValue = value; if (!this._editMode && value !== undefined) { this.field.setValue(value); } } onValueChange(event) { const value = event.detail.value; this.valueChange.emit(value); } /** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: InlineEditorComponent, deps: [{ token: i1.FormBuilder }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: InlineEditorComponent, selector: "tap-inline-editor", inputs: { editMode: "editMode", editable: "editable", unitValue: "unitValue", inputOptions: "inputOptions", valueStyle: "valueStyle", value: "value", forceValue: "forceValue", showLoader: "showLoader", fieldError: "fieldError" }, outputs: { onSubmit: "onSubmit", valueChange: "valueChange", editModeChange: "editModeChange" }, viewQueries: [{ propertyName: "inputView", first: true, predicate: ["inputView"], descendants: true }], ngImport: i0, template: "<ng-container [formGroup]=\"form\">\n <ion-label\n (click)=\"onValueClick()\"\n *ngIf=\"!_editMode\"\n [class]=\"editable ? 'cell editor-value' : 'editor-value'\"\n >\n <span [ngStyle]=\"valueStyle\">{{\n currentValue | valueOrDefault: '---'\n }}</span>\n <span\n *ngIf=\"unitValue\"\n [ngStyle]=\"valueStyle\"\n class=\"variable-unit ion-margin-start\"\n >{{ unitValue }}</span\n >\n </ion-label>\n <ion-input\n #inputView\n (keydown.enter)=\"onValueSubmit()\"\n (ionChange)=\"onValueChange($event)\"\n [class]=\"editable ? 'cell' : 'editor-value'\"\n [ngStyle]=\"valueStyle\"\n (keydown.escape)=\"onCancelEdit()\"\n *ngIf=\"_editMode\"\n [max]=\"inputOptions.max\"\n [maxlength]=\"inputOptions.maxLength\"\n [min]=\"inputOptions.min\"\n [minlength]=\"inputOptions.minLength\"\n [pattern]=\"inputOptions.pattern\"\n [placeholder]=\"inputOptions.placeholder\"\n [type]=\"inputOptions.type\"\n appAutoFocus\n class=\"cellInput\"\n [formControlName]=\"'field'\"\n ></ion-input>\n</ng-container>\n", styles: [".cell{width:fit-content}.cell:hover{cursor:pointer}.cellInput{width:fit-content;height:100%}ion-select{width:100%;min-width:150px;max-width:unset}:host ::ng-deep .toggle-disabled,:host ::ng-deep .select-disabled{opacity:.7}.spanToggle{vertical-align:top;margin:8px;font-weight:550}:host ::ng-deep ion-buttons[slot=end]{margin-inline-start:0px}:host ::ng-deep .native-input.sc-ion-input-md{padding-bottom:0!important;padding-top:0!important}ion-label{visibility:inherit}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i3.IonInput, selector: "ion-input", inputs: ["autocapitalize", "autocomplete", "autocorrect", "autofocus", "clearInput", "clearInputIcon", "clearOnEdit", "color", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "max", "maxlength", "min", "minlength", "mode", "multiple", "name", "pattern", "placeholder", "readonly", "required", "shape", "spellcheck", "step", "type", "value"] }, { kind: "component", type: i3.IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "directive", type: i3.TextValueAccessor, selector: "ion-input:not([type=number]),ion-input-otp[type=text],ion-textarea,ion-searchbar" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.MinLengthValidator, selector: "[minlength][formControlName],[minlength][formControl],[minlength][ngModel]", inputs: ["minlength"] }, { kind: "directive", type: i1.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i1.PatternValidator, selector: "[pattern][formControlName],[pattern][formControl],[pattern][ngModel]", inputs: ["pattern"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "pipe", type: i4.ValueOrDefaultPipe, name: "valueOrDefault" }] }); } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: InlineEditorComponent, decorators: [{ type: Component, args: [{ selector: 'tap-inline-editor', template: "<ng-container [formGroup]=\"form\">\n <ion-label\n (click)=\"onValueClick()\"\n *ngIf=\"!_editMode\"\n [class]=\"editable ? 'cell editor-value' : 'editor-value'\"\n >\n <span [ngStyle]=\"valueStyle\">{{\n currentValue | valueOrDefault: '---'\n }}</span>\n <span\n *ngIf=\"unitValue\"\n [ngStyle]=\"valueStyle\"\n class=\"variable-unit ion-margin-start\"\n >{{ unitValue }}</span\n >\n </ion-label>\n <ion-input\n #inputView\n (keydown.enter)=\"onValueSubmit()\"\n (ionChange)=\"onValueChange($event)\"\n [class]=\"editable ? 'cell' : 'editor-value'\"\n [ngStyle]=\"valueStyle\"\n (keydown.escape)=\"onCancelEdit()\"\n *ngIf=\"_editMode\"\n [max]=\"inputOptions.max\"\n [maxlength]=\"inputOptions.maxLength\"\n [min]=\"inputOptions.min\"\n [minlength]=\"inputOptions.minLength\"\n [pattern]=\"inputOptions.pattern\"\n [placeholder]=\"inputOptions.placeholder\"\n [type]=\"inputOptions.type\"\n appAutoFocus\n class=\"cellInput\"\n [formControlName]=\"'field'\"\n ></ion-input>\n</ng-container>\n", styles: [".cell{width:fit-content}.cell:hover{cursor:pointer}.cellInput{width:fit-content;height:100%}ion-select{width:100%;min-width:150px;max-width:unset}:host ::ng-deep .toggle-disabled,:host ::ng-deep .select-disabled{opacity:.7}.spanToggle{vertical-align:top;margin:8px;font-weight:550}:host ::ng-deep ion-buttons[slot=end]{margin-inline-start:0px}:host ::ng-deep .native-input.sc-ion-input-md{padding-bottom:0!important;padding-top:0!important}ion-label{visibility:inherit}\n"] }] }], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i0.ChangeDetectorRef }], propDecorators: { onSubmit: [{ type: Output }], valueChange: [{ type: Output }], editMode: [{ type: Input }], editModeChange: [{ type: Output }], editable: [{ type: Input }], unitValue: [{ type: Input }], inputOptions: [{ type: Input }], valueStyle: [{ type: Input }], value: [{ type: Input }], forceValue: [{ type: Input }], showLoader: [{ type: Input }], fieldError: [{ type: Input }], inputView: [{ type: ViewChild, args: ['inputView'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"inline-editor.component.js","sourceRoot":"","sources":["../../../../../../projects/iotize-ionic/src/lib/inline-editor/inline-editor.component.ts","../../../../../../projects/iotize-ionic/src/lib/inline-editor/inline-editor.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  Output,\n  ViewChild,\n} from '@angular/core';\nimport {\n  AbstractControl,\n  FormBuilder,\n  FormGroup,\n  Validators,\n} from '@angular/forms';\nimport { IonInput } from '@ionic/angular';\nimport { Observable, Subject, Subscription } from 'rxjs';\nimport { debug } from '../logger';\nimport { getFormFieldOrError } from '../utility';\n\nconst TAG = 'InlineEditorComponent';\n\nexport namespace InlineEditorComponent {\n  export type InputOptions =\n    | ToggleOptions\n    | NumberOptions\n    | PasswordOptions\n    | TextOptions;\n\n  export interface ToggleOptions {\n    type: 'toggle';\n  }\n\n  export interface NumberOptions {\n    min?: number;\n    max?: number;\n    type: 'number';\n  }\n\n  export interface PasswordOptions {\n    pattern?: RegExp;\n    minLength?: number;\n    maxLength?: number;\n    type: 'password';\n  }\n\n  export interface TextOptions {\n    pattern?: RegExp;\n    minLength?: number;\n    maxLength?: number;\n    placeholder?: string;\n    type: 'text';\n  }\n\n  export namespace SelectOptions {\n    export type Options = {\n      key: any;\n      text: string;\n    }[];\n  }\n}\n\n@Component({\n  selector: 'tap-inline-editor',\n  templateUrl: './inline-editor.component.html',\n  styleUrls: ['./inline-editor.component.scss'],\n})\nexport class InlineEditorComponent<ValueType = any> implements OnDestroy {\n  /**\n   * Emitter when form is submit\n   */\n  @Output() onSubmit: EventEmitter<{\n    target: { value: string };\n  }> = new EventEmitter();\n\n  @Output() valueChange = new EventEmitter<string | number>();\n\n  /**\n   * Enable/disable edit mode\n   */\n  @Input() set editMode(v: boolean) {\n    this._editMode = v;\n    this.field.setValue(this.currentValue);\n    if (v) {\n      this.enableEditMode();\n    }\n  }\n\n  @Output() editModeChange = new EventEmitter<boolean>();\n\n  /**\n   * True if field can be edited\n   */\n  @Input() editable = false;\n\n  @Input() unitValue?: string;\n\n  /**\n   * Input option\n   * You can defined how input will be rendered: text, number, toggle, select, ...\n   */\n  @Input() inputOptions: InlineEditorComponent.InputOptions | any = {\n    type: 'text',\n  };\n\n  /**\n   * Additional style for value\n   */\n  @Input() valueStyle?: string;\n\n  /**\n   * Set observable to subscribe to get value updates\n   */\n  @Input() set value(obs: Observable<ValueType> | ValueType) {\n    this.valueSubscription?.unsubscribe();\n    if (obs && obs instanceof Observable) {\n      this.valueSubscription = obs.subscribe((value) => {\n        this.updateValue(value);\n      });\n    } else if (obs !== undefined) {\n      this.updateValue(obs as ValueType);\n    }\n  }\n\n  /**\n   * Set value snapshot\n   */\n  @Input() set forceValue(value: any) {\n    this.updateValue(value);\n  }\n\n  /**\n   * Enable/disable loader when submitting a value\n   */\n  @Input() showLoader = true;\n\n  /**\n   * Set field error message\n   */\n  @Input() set fieldError(err: Error) {\n    if (err) {\n      debug(TAG, 'Set field error', err);\n      this.field.setErrors({\n        submit: err.message,\n      });\n    } else {\n      this.field.setErrors(null);\n    }\n  }\n\n  @ViewChild('inputView') inputView!: IonInput;\n\n  private originalValue?: ValueType;\n\n  public form: FormGroup;\n  public _editMode = false;\n\n  currentValue: any;\n  valueSubscription?: Subscription;\n\n  field: AbstractControl;\n\n  private destroyed = new Subject<void>();\n\n  constructor(\n    formBuilder: FormBuilder,\n    private changeDetectorRef: ChangeDetectorRef\n  ) {\n    this.form = formBuilder.group({\n      field: ['', Validators.required],\n    });\n    this.field = getFormFieldOrError(this.form, 'field');\n  }\n\n  ngOnDestroy(): void {\n    this.destroyed.next();\n    this.destroyed.complete();\n    if (this.valueSubscription) {\n      this.valueSubscription.unsubscribe();\n    }\n  }\n\n  public onValueSubmit() {\n    const value = this.field.value;\n    if (this.field.errors) {\n      return;\n    }\n    this.onSubmit.emit(value);\n  }\n\n  isEdited() {\n    return this.field.value !== this.originalValue;\n  }\n\n  restoreValue() {\n    this.setFieldValue(this.originalValue);\n  }\n\n  setFieldValue(value: ValueType | undefined) {\n    // let formattedValue = this.formatter()\n    // this.inputFormatter ? this._info.editFormatter.write(value) : value;\n    this.field.setValue(value || '');\n  }\n\n  async onValueClick() {\n    if (this.editable) {\n      await this.enableEditMode();\n      this.editModeChange.next(true);\n    }\n  }\n\n  onCancelEdit() {\n    this._editMode = false;\n    this.editModeChange.next(false);\n  }\n\n  async enableEditMode() {\n    this._editMode = true;\n    await this.setInputFocus();\n  }\n\n  private async setInputFocus() {\n    this.changeDetectorRef.detectChanges();\n    await this.inputView.setFocus();\n  }\n\n  private updateValue(value: ValueType) {\n    this.currentValue = value;\n    if (!this._editMode && value !== undefined) {\n      this.field.setValue(value);\n    }\n  }\n\n  onValueChange(event: CustomEvent<{ value: string }>) {\n    const value = event.detail.value;\n    this.valueChange.emit(value);\n  }\n}\n","<ng-container [formGroup]=\"form\">\n  <ion-label\n    (click)=\"onValueClick()\"\n    *ngIf=\"!_editMode\"\n    [class]=\"editable ? 'cell editor-value' : 'editor-value'\"\n  >\n    <span [ngStyle]=\"valueStyle\">{{\n      currentValue | valueOrDefault: '---'\n    }}</span>\n    <span\n      *ngIf=\"unitValue\"\n      [ngStyle]=\"valueStyle\"\n      class=\"variable-unit ion-margin-start\"\n      >{{ unitValue }}</span\n    >\n  </ion-label>\n  <ion-input\n    #inputView\n    (keydown.enter)=\"onValueSubmit()\"\n    (ionChange)=\"onValueChange($event)\"\n    [class]=\"editable ? 'cell' : 'editor-value'\"\n    [ngStyle]=\"valueStyle\"\n    (keydown.escape)=\"onCancelEdit()\"\n    *ngIf=\"_editMode\"\n    [max]=\"inputOptions.max\"\n    [maxlength]=\"inputOptions.maxLength\"\n    [min]=\"inputOptions.min\"\n    [minlength]=\"inputOptions.minLength\"\n    [pattern]=\"inputOptions.pattern\"\n    [placeholder]=\"inputOptions.placeholder\"\n    [type]=\"inputOptions.type\"\n    appAutoFocus\n    class=\"cellInput\"\n    [formControlName]=\"'field'\"\n  ></ion-input>\n</ng-container>\n"]}