@data-cafe/datagrid
Version:
A very generic datagrid component for data-café applications
126 lines • 23.1 kB
JavaScript
import { __decorate } from "tslib";
/*
* data·café
* Copyright (c) 2021-2022 Data Terrae
* This program is under the terms of the GNU Affero General Public License version 3
* The full license information can be found in LICENSE in the root directory of this project.
*/
import { Component, EventEmitter, Input, Output, } from '@angular/core';
import { FormControl } from '@angular/forms';
import { UntilDestroy } from '@ngneat/until-destroy';
import { ICONS } from '../constant';
import * as i0 from "@angular/core";
import * as i1 from "@ngneat/edit-in-place";
import * as i2 from "@angular/common";
import * as i3 from "@angular/forms";
import * as i4 from "@clr/angular";
let CellEditableComponent = class CellEditableComponent {
constructor(renderer, element) {
this.renderer = renderer;
this.element = element;
this.ICON = ICONS;
/**
* Main item to edit.
* It could be null, meaning no value.
* The `undefined` value is not allowed here; please use `null` instead.
*/
this.item = null;
/**
* Allow edition
*/
this.enable = true;
/**
* Specify some extra validators
*/
this.validators = [];
/**
* call on save:
* - button validate
* - enter key
*/
this.save = new EventEmitter();
/**
* Dispatch a event indicating whenever this component is in edition mode.
*/
this.editing$ = new EventEmitter();
this.formControl = new FormControl();
}
ngOnInit() {
// Styling parent cell <td>
const cellElement = this.element.nativeElement.closest('clr-dg-cell');
this.renderer.addClass(cellElement, 'has-edit');
}
getLabel(value) {
if (!value) {
return undefined;
}
else if (['string', 'number'].includes(typeof value)) {
return `${value}`;
}
else if (this.itemToLabel) {
return this.itemToLabel(value);
}
else if ('label' in value) {
return value.label;
}
else if ('name' in value) {
return value.name;
}
else if ('title' in value) {
return value.title;
}
else {
return '' + value;
}
}
onEditableChange(mode) {
if (mode === 'edit') {
this.formControl.setValidators(this.validators);
this.formControl.setValue(this.item);
}
this.editing$.emit(mode === 'edit');
}
trim(value) {
if (typeof value === 'string') {
return value.trim();
}
else {
return value;
}
}
onEdit() {
if (this.formControl.valid) {
this.save.emit({
previous: this.item ?? null,
current: this.trim(this.formControl.value) ?? null,
});
}
}
};
CellEditableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: CellEditableComponent, deps: [{ token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
CellEditableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.1.1", type: CellEditableComponent, selector: "dt-cell-editable", inputs: { item: "item", fallbackLabel: "fallbackLabel", altValues: "altValues", itemToLabel: "itemToLabel", enable: "enable", validators: "validators" }, outputs: { save: "save", editing$: "editing$" }, ngImport: i0, template: "<!--\n ~ data\u00B7caf\u00E9\n ~ Copyright (c) 2021-2022 Data Terrae\n ~ This program is under the terms of the GNU Affero General Public License version 3\n ~ The full license information can be found in LICENSE in the root directory of this project.\n -->\n\n<editable (modeChange)=\"onEditableChange($event)\"\n (save)=\"onEdit()\"\n [class.can-edit]=\"enable\"\n [class.is-editing]=\"editing$ | async\"\n [enabled]=\"enable\">\n <ng-template viewMode>\n <span [class.missing]=\"!item && fallbackLabel\"\n [innerText]=\"getLabel(item) ?? fallbackLabel ?? ''\"></span>\n </ng-template>\n <ng-template editMode>\n\n <input *ngIf=\"!altValues\"\n [formControl]=\"formControl\"\n clrInput\n editableFocusable editableOnEnter editableOnEscape trim=\"blur\"/>\n\n <select *ngIf=\"altValues\"\n [formControl]=\"formControl\"\n clrSelect\n editableFocusable editableOnEnter editableOnEscape>\n <option [label]=\"fallbackLabel\" [ngValue]=\"null\" class=\"missing\"></option>\n <option *ngFor=\"let alt of altValues\" [label]=\"getLabel(alt)\" [ngValue]=\"alt\"></option>\n </select>\n\n <div class=\"btn-group btn-sm btn-icon\">\n <button class=\"btn btn-success\" editableOnSave>\n <cds-icon [attr.shape]=\"ICON.VALID\"></cds-icon>\n </button>\n <button class=\"btn btn-danger\" editableOnCancel>\n <cds-icon [attr.shape]=\"ICON.CANCEL\"></cds-icon>\n </button>\n </div>\n </ng-template>\n</editable>\n", styles: ["@charset \"UTF-8\";/*!\n * data\u00B7caf\u00E9\n * Copyright (c) 2021-2022 Data Terrae\n * This program is under the terms of the GNU Affero General Public License version 3\n * The full license information can be found in LICENSE in the root directory of this project.\n */:host{display:contents}:host editable{height:2rem;display:flex;flex-direction:row;align-items:center;justify-content:space-between}:host editable.can-edit{cursor:pointer}:host editable:not(.is-editing){flex-flow:wrap;padding:0 .6rem;overflow-wrap:break-word}:host editable.is-editing{flex-flow:nowrap;padding:0 .3rem}:host editable ::ng-deep .clr-form-control{display:contents}:host editable ::ng-deep .clr-form-control .clr-control-container{display:contents;margin:0}:host editable ::ng-deep .clr-form-control .clr-control-container .clr-input-wrapper,:host editable ::ng-deep .clr-form-control .clr-control-container .clr-select-wrapper{display:contents}:host editable ::ng-deep .clr-form-control .clr-control-container .clr-input-wrapper input,:host editable ::ng-deep .clr-form-control .clr-control-container .clr-input-wrapper select,:host editable ::ng-deep .clr-form-control .clr-control-container .clr-select-wrapper input,:host editable ::ng-deep .clr-form-control .clr-control-container .clr-select-wrapper select{width:calc(100% - 4.2rem);min-width:unset;max-width:unset;flex-grow:1;flex-shrink:1}:host editable ::ng-deep .btn-group{min-width:3.8rem;flex-grow:0;flex-shrink:0}:host editable ::ng-deep *+.btn-group{margin:0 0 0 .4rem}\n"], components: [{ type: i1.EditableComponent, selector: "editable", inputs: ["enabled", "openBindingEvent", "closeBindingEvent"], outputs: ["save", "cancel", "modeChange"] }], directives: [{ type: i1.ViewModeDirective, selector: "[viewMode]" }, { type: i1.EditModeDirective, selector: "[editMode]" }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i4.ClrInput, selector: "[clrInput]" }, { type: i1.EditableFocusDirective, selector: "[editableFocusable]" }, { type: i1.EditableOnEnterDirective, selector: "[editableOnEnter]" }, { type: i1.EditableOnEscapeDirective, selector: "[editableOnEscape]" }, { type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { type: i3.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { type: i4.ClrSelect, selector: "[clrSelect]" }, { type: i3.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i3.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.EditableSaveDirective, selector: "[editableOnSave]", inputs: ["saveEvent"] }, { type: i4.CdsIconCustomTag, selector: "cds-icon" }, { type: i1.EditableCancelDirective, selector: "[editableOnCancel]", inputs: ["cancelEvent"] }], pipes: { "async": i2.AsyncPipe } });
CellEditableComponent = __decorate([
UntilDestroy()
], CellEditableComponent);
export { CellEditableComponent };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: CellEditableComponent, decorators: [{
type: Component,
args: [{ selector: 'dt-cell-editable', template: "<!--\n ~ data\u00B7caf\u00E9\n ~ Copyright (c) 2021-2022 Data Terrae\n ~ This program is under the terms of the GNU Affero General Public License version 3\n ~ The full license information can be found in LICENSE in the root directory of this project.\n -->\n\n<editable (modeChange)=\"onEditableChange($event)\"\n (save)=\"onEdit()\"\n [class.can-edit]=\"enable\"\n [class.is-editing]=\"editing$ | async\"\n [enabled]=\"enable\">\n <ng-template viewMode>\n <span [class.missing]=\"!item && fallbackLabel\"\n [innerText]=\"getLabel(item) ?? fallbackLabel ?? ''\"></span>\n </ng-template>\n <ng-template editMode>\n\n <input *ngIf=\"!altValues\"\n [formControl]=\"formControl\"\n clrInput\n editableFocusable editableOnEnter editableOnEscape trim=\"blur\"/>\n\n <select *ngIf=\"altValues\"\n [formControl]=\"formControl\"\n clrSelect\n editableFocusable editableOnEnter editableOnEscape>\n <option [label]=\"fallbackLabel\" [ngValue]=\"null\" class=\"missing\"></option>\n <option *ngFor=\"let alt of altValues\" [label]=\"getLabel(alt)\" [ngValue]=\"alt\"></option>\n </select>\n\n <div class=\"btn-group btn-sm btn-icon\">\n <button class=\"btn btn-success\" editableOnSave>\n <cds-icon [attr.shape]=\"ICON.VALID\"></cds-icon>\n </button>\n <button class=\"btn btn-danger\" editableOnCancel>\n <cds-icon [attr.shape]=\"ICON.CANCEL\"></cds-icon>\n </button>\n </div>\n </ng-template>\n</editable>\n", styles: ["@charset \"UTF-8\";/*!\n * data\u00B7caf\u00E9\n * Copyright (c) 2021-2022 Data Terrae\n * This program is under the terms of the GNU Affero General Public License version 3\n * The full license information can be found in LICENSE in the root directory of this project.\n */:host{display:contents}:host editable{height:2rem;display:flex;flex-direction:row;align-items:center;justify-content:space-between}:host editable.can-edit{cursor:pointer}:host editable:not(.is-editing){flex-flow:wrap;padding:0 .6rem;overflow-wrap:break-word}:host editable.is-editing{flex-flow:nowrap;padding:0 .3rem}:host editable ::ng-deep .clr-form-control{display:contents}:host editable ::ng-deep .clr-form-control .clr-control-container{display:contents;margin:0}:host editable ::ng-deep .clr-form-control .clr-control-container .clr-input-wrapper,:host editable ::ng-deep .clr-form-control .clr-control-container .clr-select-wrapper{display:contents}:host editable ::ng-deep .clr-form-control .clr-control-container .clr-input-wrapper input,:host editable ::ng-deep .clr-form-control .clr-control-container .clr-input-wrapper select,:host editable ::ng-deep .clr-form-control .clr-control-container .clr-select-wrapper input,:host editable ::ng-deep .clr-form-control .clr-control-container .clr-select-wrapper select{width:calc(100% - 4.2rem);min-width:unset;max-width:unset;flex-grow:1;flex-shrink:1}:host editable ::ng-deep .btn-group{min-width:3.8rem;flex-grow:0;flex-shrink:0}:host editable ::ng-deep *+.btn-group{margin:0 0 0 .4rem}\n"] }]
}], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i0.ElementRef }]; }, propDecorators: { item: [{
type: Input
}], fallbackLabel: [{
type: Input
}], altValues: [{
type: Input
}], itemToLabel: [{
type: Input
}], enable: [{
type: Input
}], validators: [{
type: Input
}], save: [{
type: Output
}], editing$: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cell-editable.component.js","sourceRoot":"","sources":["../../../../../projects/datagrid/src/lib/cell-editable/cell-editable.component.ts","../../../../../projects/datagrid/src/lib/cell-editable/cell-editable.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * data·café\n * Copyright (c) 2021-2022 Data Terrae\n * This program is under the terms of the GNU Affero General Public License version 3\n * The full license information can be found in LICENSE in the root directory of this project.\n */\n\nimport {\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnInit,\n  Output,\n  Renderer2,\n} from '@angular/core';\nimport { FormControl, ValidatorFn } from '@angular/forms';\nimport { UntilDestroy } from '@ngneat/until-destroy';\nimport { ICONS } from '../constant';\n\nexport interface Result<T> {\n  previous: T | null,\n  current: T | null,\n}\n\n@UntilDestroy()\n@Component({\n  selector: 'dt-cell-editable',\n  templateUrl: './cell-editable.component.html',\n  styleUrls: ['./cell-editable.component.scss']\n})\nexport class CellEditableComponent<T> implements OnInit {\n\n  readonly ICON = ICONS;\n\n  /**\n   * Main item to edit.\n   * It could be null, meaning no value.\n   * The `undefined` value is not allowed here; please use `null` instead.\n   */\n  @Input() item: T | null = null;\n  /**\n   * Fallback label used when item is null or falsy, or the property is missing.\n   *\n   * This is optional.\n   */\n  @Input() fallbackLabel?: string;\n  /**\n   * This list is a range of allowed values.\n   * It displays a select element instead of a simple input.\n   *\n   * This is optional.\n   */\n  @Input() altValues?: Array<T>;\n  /**\n   * Converter method.\n   *\n   * This is optional only if:\n   * - the item is a string\n   * - the item contains a property `label`, `name` or `title`\n   */\n  @Input() itemToLabel?: (t: T) => string;\n\n  /**\n   * Allow edition\n   */\n  @Input() enable = true;\n\n  /**\n   * Specify some extra validators\n   */\n  @Input() validators: Array<ValidatorFn> = [];\n\n  /**\n   * call on save:\n   * - button validate\n   * - enter key\n   */\n  @Output() save: EventEmitter<Result<T>> = new EventEmitter<Result<T>>();\n\n  /**\n   * Dispatch a event indicating whenever this component is in edition mode.\n   */\n  @Output() editing$: EventEmitter<boolean> = new EventEmitter<boolean>();\n\n  formControl = new FormControl();\n\n  constructor(readonly renderer: Renderer2,\n              readonly element: ElementRef) {\n  }\n\n  ngOnInit(): void {\n    // Styling parent cell <td>\n    const cellElement = this.element.nativeElement.closest('clr-dg-cell');\n    this.renderer.addClass(cellElement, 'has-edit');\n  }\n\n  getLabel(value: T | null): string | undefined {\n    if (!value) {\n      return undefined;\n    } else if (['string', 'number'].includes(typeof value)) {\n      return `${value}`;\n    } else if (this.itemToLabel) {\n      return this.itemToLabel(value);\n    } else if ('label' in value) {\n      return (value as unknown as { label: string }).label;\n    } else if ('name' in value) {\n      return (value as unknown as { name: string }).name;\n    } else if ('title' in value) {\n      return (value as unknown as { title: string }).title;\n    } else {\n      return '' + value;\n    }\n  }\n\n  onEditableChange(mode: 'view' | 'edit'): void {\n    if (mode === 'edit') {\n      this.formControl.setValidators(this.validators);\n      this.formControl.setValue(this.item);\n    }\n    this.editing$.emit(mode === 'edit');\n  }\n\n  trim(value: T | null): T | null {\n    if (typeof value === 'string') {\n      return value.trim() as unknown as T;\n    } else {\n      return value;\n    }\n  }\n\n  onEdit() {\n    if (this.formControl.valid) {\n      this.save.emit({\n        previous: this.item ?? null,\n        current: this.trim(this.formControl.value) ?? null,\n      });\n    }\n  }\n}\n","<!--\n  ~ data·café\n  ~ Copyright (c) 2021-2022 Data Terrae\n  ~ This program is under the terms of the GNU Affero General Public License version 3\n  ~ The full license information can be found in LICENSE in the root directory of this project.\n  -->\n\n<editable (modeChange)=\"onEditableChange($event)\"\n          (save)=\"onEdit()\"\n          [class.can-edit]=\"enable\"\n          [class.is-editing]=\"editing$ | async\"\n          [enabled]=\"enable\">\n  <ng-template viewMode>\n    <span [class.missing]=\"!item && fallbackLabel\"\n          [innerText]=\"getLabel(item) ?? fallbackLabel ?? ''\"></span>\n  </ng-template>\n  <ng-template editMode>\n\n    <input *ngIf=\"!altValues\"\n           [formControl]=\"formControl\"\n           clrInput\n           editableFocusable editableOnEnter editableOnEscape trim=\"blur\"/>\n\n    <select *ngIf=\"altValues\"\n            [formControl]=\"formControl\"\n            clrSelect\n            editableFocusable editableOnEnter editableOnEscape>\n      <option [label]=\"fallbackLabel\" [ngValue]=\"null\" class=\"missing\"></option>\n      <option *ngFor=\"let alt of altValues\" [label]=\"getLabel(alt)\" [ngValue]=\"alt\"></option>\n    </select>\n\n    <div class=\"btn-group btn-sm btn-icon\">\n      <button class=\"btn btn-success\" editableOnSave>\n        <cds-icon [attr.shape]=\"ICON.VALID\"></cds-icon>\n      </button>\n      <button class=\"btn btn-danger\" editableOnCancel>\n        <cds-icon [attr.shape]=\"ICON.CANCEL\"></cds-icon>\n      </button>\n    </div>\n  </ng-template>\n</editable>\n"]}