@c80/ui 0.0.1 → 1.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/index.mjs +3 -2
- package/esm2022/lib/icon/icon.component.mjs +121 -0
- package/esm2022/lib/table/table.component.mjs +237 -0
- package/index.d.ts +2 -1
- package/lib/icon/icon.component.d.ts +31 -0
- package/lib/table/table.component.d.ts +60 -0
- package/package.json +5 -3
- package/esm2022/lib/ui/ui.component.mjs +0 -12
- package/lib/ui/ui.component.d.ts +0 -5
package/esm2022/index.mjs
CHANGED
|
@@ -1,2 +1,3 @@
|
|
|
1
|
-
export * from './lib/
|
|
2
|
-
|
|
1
|
+
export * from './lib/table/table.component';
|
|
2
|
+
export * from './lib/icon/icon.component';
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWJzL3VpL3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLDZCQUE2QixDQUFDO0FBQzVDLGNBQWMsMkJBQTJCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2xpYi90YWJsZS90YWJsZS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvaWNvbi9pY29uLmNvbXBvbmVudCc7XG4iXX0=
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ViewChild } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
export class C80IconComponent {
|
|
6
|
+
/** Tamaño base del icono (en px) */
|
|
7
|
+
get iconSize() {
|
|
8
|
+
return 24 * this.size;
|
|
9
|
+
}
|
|
10
|
+
icon = 'check';
|
|
11
|
+
color = 'primary';
|
|
12
|
+
disabled = false;
|
|
13
|
+
size = 1;
|
|
14
|
+
/** Si es true, renderiza como <button> nativo, si es false solo como icono */
|
|
15
|
+
_button = false;
|
|
16
|
+
set button(val) {
|
|
17
|
+
// Permite usar 'button' a secas, cualquier valor que no sea null/undefined/false es true
|
|
18
|
+
this._button = val !== null && val !== undefined && val !== false && val !== 'false';
|
|
19
|
+
}
|
|
20
|
+
get button() {
|
|
21
|
+
return this._button;
|
|
22
|
+
}
|
|
23
|
+
/** Tipo de botón nativo (solo si button=true) */
|
|
24
|
+
type = 'button';
|
|
25
|
+
/** Output para click (solo si button=true) */
|
|
26
|
+
iconClick = new EventEmitter();
|
|
27
|
+
checkIconTpl;
|
|
28
|
+
cancelIconTpl;
|
|
29
|
+
editIconTpl;
|
|
30
|
+
deleteIconTpl;
|
|
31
|
+
addIconTpl;
|
|
32
|
+
defaultIconTpl;
|
|
33
|
+
getIconTemplate() {
|
|
34
|
+
switch (this.icon) {
|
|
35
|
+
case 'check': return this.checkIconTpl;
|
|
36
|
+
case 'cancel': return this.cancelIconTpl;
|
|
37
|
+
case 'edit': return this.editIconTpl;
|
|
38
|
+
case 'delete': return this.deleteIconTpl;
|
|
39
|
+
case 'add': return this.addIconTpl;
|
|
40
|
+
default: return this.defaultIconTpl;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
get iconColor() {
|
|
44
|
+
if (this.disabled) {
|
|
45
|
+
return '#bdbdbd'; // gris deshabilitado
|
|
46
|
+
}
|
|
47
|
+
const palettes = {
|
|
48
|
+
check: {
|
|
49
|
+
primary: '#00234bad',
|
|
50
|
+
secondary: '#6c757d',
|
|
51
|
+
warn: '#e53935',
|
|
52
|
+
},
|
|
53
|
+
cancel: {
|
|
54
|
+
primary: '#00234bad',
|
|
55
|
+
secondary: '#6c757d',
|
|
56
|
+
warn: '#e53935',
|
|
57
|
+
},
|
|
58
|
+
edit: {
|
|
59
|
+
primary: '#00234bad',
|
|
60
|
+
secondary: '#6c757d',
|
|
61
|
+
warn: '#e53935',
|
|
62
|
+
},
|
|
63
|
+
delete: {
|
|
64
|
+
primary: '#00234bad',
|
|
65
|
+
secondary: '#6c757d',
|
|
66
|
+
warn: '#e53935',
|
|
67
|
+
},
|
|
68
|
+
add: {
|
|
69
|
+
primary: '#00234bad',
|
|
70
|
+
secondary: '#6c757d',
|
|
71
|
+
warn: '#e53935',
|
|
72
|
+
}
|
|
73
|
+
};
|
|
74
|
+
return palettes[this.icon]?.[this.color] ?? '#222';
|
|
75
|
+
}
|
|
76
|
+
// Handler para click en modo button
|
|
77
|
+
onButtonClick(event) {
|
|
78
|
+
if (!this.disabled) {
|
|
79
|
+
this.iconClick.emit(event);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: C80IconComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
83
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: C80IconComponent, isStandalone: true, selector: "c80-icon", inputs: { icon: "icon", color: "color", disabled: "disabled", size: "size", button: "button", type: "type" }, outputs: { iconClick: "iconClick" }, viewQueries: [{ propertyName: "checkIconTpl", first: true, predicate: ["checkIcon"], descendants: true, static: true }, { propertyName: "cancelIconTpl", first: true, predicate: ["cancelIcon"], descendants: true, static: true }, { propertyName: "editIconTpl", first: true, predicate: ["editIcon"], descendants: true, static: true }, { propertyName: "deleteIconTpl", first: true, predicate: ["deleteIcon"], descendants: true, static: true }, { propertyName: "addIconTpl", first: true, predicate: ["addIcon"], descendants: true, static: true }, { propertyName: "defaultIconTpl", first: true, predicate: ["defaultIcon"], descendants: true, static: true }], ngImport: i0, template: "@if (button) {\r\n<button type=\"{{type}}\" [disabled]=\"disabled\" class=\"icon-button\" [style.width.px]=\"iconSize + 8\"\r\n [style.height.px]=\"iconSize + 8\" (click)=\"onButtonClick($event)\">\r\n <ng-container *ngTemplateOutlet=\"getIconTemplate()\"></ng-container>\r\n</button>\r\n} @else {\r\n<span class=\"icon-span\" [style.width.px]=\"iconSize\" [style.height.px]=\"iconSize\">\r\n <ng-container *ngTemplateOutlet=\"getIconTemplate()\"></ng-container>\r\n</span>\r\n}\r\n\r\n<ng-template #checkIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <path d=\"M5 13l4 4L19 7\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #cancelIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <path d=\"M6 6l12 12M6 18L18 6\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #editIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <rect x=\"5\" y=\"19\" width=\"14\" height=\"2\" rx=\"1\" [attr.fill]=\"iconColor\" />\r\n <path d=\"M16.5 3.5a2.121 2.121 0 0 1 3 3L7 19l-4 1 1-4 12.5-12.5z\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #deleteIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <path d=\"M6 19a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z\" [attr.fill]=\"iconColor\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #addIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <path d=\"M12 5v14M5 12h14\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #defaultIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <circle cx=\"12\" cy=\"12\" r=\"10\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" />\r\n </svg>\r\n</ng-template>", styles: [":host .icon-button{display:inline-flex;align-items:center;justify-content:center;border:none;outline:none;background:transparent;border-radius:50%;min-width:0;min-height:0;padding:4px;cursor:pointer;transition:background .2s;box-sizing:border-box}:host .icon-button:focus-visible{outline:2px solid #1976d2;outline-offset:2px}:host .icon-button:hover:not(:disabled){background:#1976d214}:host .icon-button:active:not(:disabled){background:#1976d229}:host .icon-button:disabled{opacity:.5;cursor:default;background:transparent}:host .icon-span,:host .icon-button{margin-left:4px;margin-right:4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }] });
|
|
84
|
+
}
|
|
85
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: C80IconComponent, decorators: [{
|
|
86
|
+
type: Component,
|
|
87
|
+
args: [{ selector: 'c80-icon', standalone: true, imports: [CommonModule], template: "@if (button) {\r\n<button type=\"{{type}}\" [disabled]=\"disabled\" class=\"icon-button\" [style.width.px]=\"iconSize + 8\"\r\n [style.height.px]=\"iconSize + 8\" (click)=\"onButtonClick($event)\">\r\n <ng-container *ngTemplateOutlet=\"getIconTemplate()\"></ng-container>\r\n</button>\r\n} @else {\r\n<span class=\"icon-span\" [style.width.px]=\"iconSize\" [style.height.px]=\"iconSize\">\r\n <ng-container *ngTemplateOutlet=\"getIconTemplate()\"></ng-container>\r\n</span>\r\n}\r\n\r\n<ng-template #checkIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <path d=\"M5 13l4 4L19 7\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #cancelIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <path d=\"M6 6l12 12M6 18L18 6\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #editIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <rect x=\"5\" y=\"19\" width=\"14\" height=\"2\" rx=\"1\" [attr.fill]=\"iconColor\" />\r\n <path d=\"M16.5 3.5a2.121 2.121 0 0 1 3 3L7 19l-4 1 1-4 12.5-12.5z\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #deleteIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <path d=\"M6 19a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z\" [attr.fill]=\"iconColor\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #addIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <path d=\"M12 5v14M5 12h14\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\" />\r\n </svg>\r\n</ng-template>\r\n<ng-template #defaultIcon>\r\n <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n <circle cx=\"12\" cy=\"12\" r=\"10\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" />\r\n </svg>\r\n</ng-template>", styles: [":host .icon-button{display:inline-flex;align-items:center;justify-content:center;border:none;outline:none;background:transparent;border-radius:50%;min-width:0;min-height:0;padding:4px;cursor:pointer;transition:background .2s;box-sizing:border-box}:host .icon-button:focus-visible{outline:2px solid #1976d2;outline-offset:2px}:host .icon-button:hover:not(:disabled){background:#1976d214}:host .icon-button:active:not(:disabled){background:#1976d229}:host .icon-button:disabled{opacity:.5;cursor:default;background:transparent}:host .icon-span,:host .icon-button{margin-left:4px;margin-right:4px}\n"] }]
|
|
88
|
+
}], propDecorators: { icon: [{
|
|
89
|
+
type: Input
|
|
90
|
+
}], color: [{
|
|
91
|
+
type: Input
|
|
92
|
+
}], disabled: [{
|
|
93
|
+
type: Input
|
|
94
|
+
}], size: [{
|
|
95
|
+
type: Input
|
|
96
|
+
}], button: [{
|
|
97
|
+
type: Input
|
|
98
|
+
}], type: [{
|
|
99
|
+
type: Input
|
|
100
|
+
}], iconClick: [{
|
|
101
|
+
type: Output
|
|
102
|
+
}], checkIconTpl: [{
|
|
103
|
+
type: ViewChild,
|
|
104
|
+
args: ['checkIcon', { static: true }]
|
|
105
|
+
}], cancelIconTpl: [{
|
|
106
|
+
type: ViewChild,
|
|
107
|
+
args: ['cancelIcon', { static: true }]
|
|
108
|
+
}], editIconTpl: [{
|
|
109
|
+
type: ViewChild,
|
|
110
|
+
args: ['editIcon', { static: true }]
|
|
111
|
+
}], deleteIconTpl: [{
|
|
112
|
+
type: ViewChild,
|
|
113
|
+
args: ['deleteIcon', { static: true }]
|
|
114
|
+
}], addIconTpl: [{
|
|
115
|
+
type: ViewChild,
|
|
116
|
+
args: ['addIcon', { static: true }]
|
|
117
|
+
}], defaultIconTpl: [{
|
|
118
|
+
type: ViewChild,
|
|
119
|
+
args: ['defaultIcon', { static: true }]
|
|
120
|
+
}] } });
|
|
121
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"icon.component.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/icon/icon.component.ts","../../../../../libs/ui/src/lib/icon/icon.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, ViewChild, TemplateRef } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\n\r\nexport type Icon = 'check' | 'cancel' | 'edit' | 'delete' | 'add';\r\nexport type Color = 'primary' | 'secondary' | 'warn';\r\n\r\n@Component({\r\n  selector: 'c80-icon',\r\n  standalone: true,\r\n  imports: [CommonModule],\r\n  templateUrl: './icon.component.html',\r\n  styleUrls: ['./icon.component.scss']\r\n})\r\nexport class C80IconComponent {\r\n  /** Tamaño base del icono (en px) */\r\n  get iconSize(): number {\r\n    return 24 * this.size;\r\n  }\r\n  @Input() icon: Icon = 'check';\r\n  @Input() color: Color = 'primary';\r\n  @Input() disabled?: boolean = false;\r\n  @Input() size = 1;\r\n  /** Si es true, renderiza como <button> nativo, si es false solo como icono */\r\n  private _button = false;\r\n  @Input()\r\n  set button(val: boolean | string | null | undefined) {\r\n    // Permite usar 'button' a secas, cualquier valor que no sea null/undefined/false es true\r\n    this._button = val !== null && val !== undefined && val !== false && val !== 'false';\r\n  }\r\n  get button(): boolean {\r\n    return this._button;\r\n  }\r\n  /** Tipo de botón nativo (solo si button=true) */\r\n  @Input() type: 'button' | 'submit' | 'reset' = 'button';\r\n  /** Output para click (solo si button=true) */\r\n  @Output() iconClick = new EventEmitter<Event>();\r\n\r\n  @ViewChild('checkIcon', { static: true }) checkIconTpl!: TemplateRef<unknown>;\r\n  @ViewChild('cancelIcon', { static: true }) cancelIconTpl!: TemplateRef<unknown>;\r\n  @ViewChild('editIcon', { static: true }) editIconTpl!: TemplateRef<unknown>;\r\n  @ViewChild('deleteIcon', { static: true }) deleteIconTpl!: TemplateRef<unknown>;\r\n  @ViewChild('addIcon', { static: true }) addIconTpl!: TemplateRef<unknown>;\r\n  @ViewChild('defaultIcon', { static: true }) defaultIconTpl!: TemplateRef<unknown>;\r\n\r\n  getIconTemplate(): TemplateRef<unknown> {\r\n    switch (this.icon) {\r\n      case 'check': return this.checkIconTpl;\r\n      case 'cancel': return this.cancelIconTpl;\r\n      case 'edit': return this.editIconTpl;\r\n      case 'delete': return this.deleteIconTpl;\r\n      case 'add': return this.addIconTpl;\r\n      default: return this.defaultIconTpl;\r\n    }\r\n  }\r\n\r\n  get iconColor(): string {\r\n    if (this.disabled) {\r\n      return '#bdbdbd'; // gris deshabilitado\r\n    }\r\n    const palettes: Record<Icon, Record<Color, string>> = {\r\n      check: {\r\n        primary: '#00234bad',\r\n        secondary: '#6c757d',\r\n        warn: '#e53935',\r\n      },\r\n      cancel: {\r\n        primary: '#00234bad',\r\n        secondary: '#6c757d',\r\n        warn: '#e53935',\r\n      },\r\n      edit: {\r\n        primary: '#00234bad',\r\n        secondary: '#6c757d',\r\n        warn: '#e53935',\r\n      },\r\n      delete: {\r\n        primary: '#00234bad',\r\n        secondary: '#6c757d',\r\n        warn: '#e53935',\r\n      },\r\n      add: {\r\n        primary: '#00234bad',\r\n        secondary: '#6c757d',\r\n        warn: '#e53935',\r\n      }\r\n    };\r\n    return palettes[this.icon]?.[this.color] ?? '#222';\r\n  }\r\n\r\n  // Handler para click en modo button\r\n  onButtonClick(event: Event) {\r\n    if (!this.disabled) {\r\n      this.iconClick.emit(event);\r\n    }\r\n  }\r\n}\r\n","@if (button) {\r\n<button type=\"{{type}}\" [disabled]=\"disabled\" class=\"icon-button\" [style.width.px]=\"iconSize + 8\"\r\n  [style.height.px]=\"iconSize + 8\" (click)=\"onButtonClick($event)\">\r\n  <ng-container *ngTemplateOutlet=\"getIconTemplate()\"></ng-container>\r\n</button>\r\n} @else {\r\n<span class=\"icon-span\" [style.width.px]=\"iconSize\" [style.height.px]=\"iconSize\">\r\n  <ng-container *ngTemplateOutlet=\"getIconTemplate()\"></ng-container>\r\n</span>\r\n}\r\n\r\n<ng-template #checkIcon>\r\n  <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n    <path d=\"M5 13l4 4L19 7\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\"\r\n      stroke-linejoin=\"round\" />\r\n  </svg>\r\n</ng-template>\r\n<ng-template #cancelIcon>\r\n  <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n    <path d=\"M6 6l12 12M6 18L18 6\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\" />\r\n  </svg>\r\n</ng-template>\r\n<ng-template #editIcon>\r\n  <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n    <rect x=\"5\" y=\"19\" width=\"14\" height=\"2\" rx=\"1\" [attr.fill]=\"iconColor\" />\r\n    <path d=\"M16.5 3.5a2.121 2.121 0 0 1 3 3L7 19l-4 1 1-4 12.5-12.5z\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" />\r\n  </svg>\r\n</ng-template>\r\n<ng-template #deleteIcon>\r\n  <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n    <path d=\"M6 19a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2V7H6v12zM19 4h-3.5l-1-1h-5l-1 1H5v2h14V4z\" [attr.fill]=\"iconColor\" />\r\n  </svg>\r\n</ng-template>\r\n<ng-template #addIcon>\r\n  <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n    <path d=\"M12 5v14M5 12h14\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" stroke-linecap=\"round\" />\r\n  </svg>\r\n</ng-template>\r\n<ng-template #defaultIcon>\r\n  <svg [attr.width]=\"iconSize\" [attr.height]=\"iconSize\" viewBox=\"0 0 24 24\" fill=\"none\">\r\n    <circle cx=\"12\" cy=\"12\" r=\"10\" [attr.stroke]=\"iconColor\" stroke-width=\"2\" />\r\n  </svg>\r\n</ng-template>"]}
|
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
import { Component, Input, Output, signal, EventEmitter } from '@angular/core';
|
|
2
|
+
import { take } from 'rxjs';
|
|
3
|
+
import { MatCheckboxModule } from '@angular/material/checkbox';
|
|
4
|
+
import { C80IconComponent } from '../icon/icon.component';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@angular/material/checkbox";
|
|
7
|
+
export class C80TableComponent {
|
|
8
|
+
service;
|
|
9
|
+
columns = [];
|
|
10
|
+
errorEvent = new EventEmitter();
|
|
11
|
+
data = signal([]);
|
|
12
|
+
keys = signal([]);
|
|
13
|
+
creating = signal(false);
|
|
14
|
+
newRow = signal(null);
|
|
15
|
+
editing = signal(null); // id of row being edited
|
|
16
|
+
editRow = signal(null);
|
|
17
|
+
ngOnInit() {
|
|
18
|
+
if (!this.service)
|
|
19
|
+
return;
|
|
20
|
+
this.refresh();
|
|
21
|
+
}
|
|
22
|
+
onInput(event, key, col) {
|
|
23
|
+
if (col?.type === 'boolean') {
|
|
24
|
+
// MatCheckboxChange
|
|
25
|
+
const checked = event.checked;
|
|
26
|
+
this.updateNewRow(key, checked);
|
|
27
|
+
}
|
|
28
|
+
else {
|
|
29
|
+
const target = event.target;
|
|
30
|
+
if (target && typeof target.value === 'string') {
|
|
31
|
+
this.updateNewRow(key, target.value);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
onDelete(row) {
|
|
36
|
+
if (!this.service.delete)
|
|
37
|
+
return;
|
|
38
|
+
const id = row['id'];
|
|
39
|
+
if (typeof id !== 'number') {
|
|
40
|
+
this.errorEvent.emit('No se puede borrar: id inválido');
|
|
41
|
+
return;
|
|
42
|
+
}
|
|
43
|
+
if (confirm('¿Está seguro de que desea borrar este elemento?')) {
|
|
44
|
+
this.service.delete(id).pipe(take(1)).subscribe({
|
|
45
|
+
next: () => this.refresh(),
|
|
46
|
+
error: (err) => this.errorEvent.emit(err?.message || 'Error al borrar')
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
refresh() {
|
|
51
|
+
this.service.get().pipe(take(1)).subscribe((items) => {
|
|
52
|
+
this.data.update(() => items);
|
|
53
|
+
// Solo mostrar las columnas visibles
|
|
54
|
+
const visibleColumns = this.columns.filter(col => col?.visible !== false);
|
|
55
|
+
this.keys.update(() => visibleColumns.map(col => col.accessor));
|
|
56
|
+
});
|
|
57
|
+
}
|
|
58
|
+
startCreate() {
|
|
59
|
+
this.creating.set(true);
|
|
60
|
+
// Inicializa newRow solo con los accessors de columnas visibles
|
|
61
|
+
const row = {};
|
|
62
|
+
this.columns.filter(col => col.visible !== false).forEach(col => {
|
|
63
|
+
row[col.accessor] = '';
|
|
64
|
+
});
|
|
65
|
+
this.newRow.set(row);
|
|
66
|
+
}
|
|
67
|
+
cancelCreate() {
|
|
68
|
+
this.creating.set(false);
|
|
69
|
+
this.newRow.set(null);
|
|
70
|
+
}
|
|
71
|
+
updateNewRow(key, value) {
|
|
72
|
+
const current = this.newRow();
|
|
73
|
+
if (!current)
|
|
74
|
+
return;
|
|
75
|
+
this.newRow.set({ ...current, [key]: value });
|
|
76
|
+
}
|
|
77
|
+
saveCreate() {
|
|
78
|
+
const row = this.newRow();
|
|
79
|
+
if (!row)
|
|
80
|
+
return;
|
|
81
|
+
const visibleColumns = this.columns.filter(col => col.visible !== false);
|
|
82
|
+
const converted = visibleColumns.reduce((acc, col) => {
|
|
83
|
+
acc[col.accessor] = this.convertCellValue(row[col.accessor], col);
|
|
84
|
+
return acc;
|
|
85
|
+
}, {});
|
|
86
|
+
if (this.service.create) {
|
|
87
|
+
this.service.create(converted).pipe(take(1)).subscribe({
|
|
88
|
+
next: () => {
|
|
89
|
+
this.cancelCreate();
|
|
90
|
+
this.refresh();
|
|
91
|
+
},
|
|
92
|
+
error: (err) => {
|
|
93
|
+
this.errorEvent.emit(err?.message || 'Error al crear');
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* Converts a cell value based on column type or sample data.
|
|
100
|
+
* Handles stringification of objects for string columns.
|
|
101
|
+
*/
|
|
102
|
+
/**
|
|
103
|
+
* Converts a cell value based on column type or sample data.
|
|
104
|
+
* Delegates to type-specific helpers for clarity and maintainability.
|
|
105
|
+
*/
|
|
106
|
+
convertCellValue(value, col) {
|
|
107
|
+
if (col.type === 'boolean')
|
|
108
|
+
return this.toBoolean(value);
|
|
109
|
+
if (col.type === 'number')
|
|
110
|
+
return this.toNumber(value);
|
|
111
|
+
if (col.type === 'string')
|
|
112
|
+
return this.toStringValue(value);
|
|
113
|
+
// Fallback: use sample data if available
|
|
114
|
+
const sample = this.data()[0];
|
|
115
|
+
if (sample && col.accessor in sample) {
|
|
116
|
+
const sampleValue = sample[col.accessor];
|
|
117
|
+
if (typeof sampleValue === 'boolean')
|
|
118
|
+
return this.toBoolean(value);
|
|
119
|
+
if (typeof sampleValue === 'number')
|
|
120
|
+
return this.toNumber(value);
|
|
121
|
+
if (typeof sampleValue === 'string')
|
|
122
|
+
return this.toStringValue(value);
|
|
123
|
+
}
|
|
124
|
+
return value;
|
|
125
|
+
}
|
|
126
|
+
/** Converts value to boolean using best practices. */
|
|
127
|
+
toBoolean(value) {
|
|
128
|
+
if (typeof value === 'boolean')
|
|
129
|
+
return value;
|
|
130
|
+
if (typeof value === 'string')
|
|
131
|
+
return value.trim().toLowerCase() === 'true' || value.trim() === '1';
|
|
132
|
+
if (typeof value === 'number')
|
|
133
|
+
return value === 1;
|
|
134
|
+
return false;
|
|
135
|
+
}
|
|
136
|
+
/** Converts value to number using best practices. */
|
|
137
|
+
toNumber(value) {
|
|
138
|
+
if (typeof value === 'number')
|
|
139
|
+
return value;
|
|
140
|
+
if (typeof value === 'string')
|
|
141
|
+
return value.trim() === '' ? undefined : Number(value);
|
|
142
|
+
if (typeof value === 'boolean')
|
|
143
|
+
return value ? 1 : 0;
|
|
144
|
+
return undefined;
|
|
145
|
+
}
|
|
146
|
+
/** Converts value to string using best practices, always stringifies objects. */
|
|
147
|
+
toStringValue(value) {
|
|
148
|
+
if (value == null)
|
|
149
|
+
return '';
|
|
150
|
+
if (typeof value === 'string')
|
|
151
|
+
return value;
|
|
152
|
+
if (typeof value === 'number' || typeof value === 'boolean')
|
|
153
|
+
return String(value);
|
|
154
|
+
if (typeof value === 'object') {
|
|
155
|
+
try {
|
|
156
|
+
return JSON.stringify(value);
|
|
157
|
+
}
|
|
158
|
+
catch {
|
|
159
|
+
return '[object Object]';
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
// For functions, symbols, undefined, etc., return empty string
|
|
163
|
+
return '';
|
|
164
|
+
}
|
|
165
|
+
onEdit(row) {
|
|
166
|
+
this.editing.set(row['id']);
|
|
167
|
+
const edit = {};
|
|
168
|
+
this.columns.filter(col => col.visible !== false).forEach(col => {
|
|
169
|
+
edit[col.accessor] = row[col.accessor];
|
|
170
|
+
});
|
|
171
|
+
this.editRow.set(edit);
|
|
172
|
+
}
|
|
173
|
+
cancelEdit() {
|
|
174
|
+
this.editing.set(null);
|
|
175
|
+
this.editRow.set(null);
|
|
176
|
+
}
|
|
177
|
+
onEditInput(event, key, col) {
|
|
178
|
+
const current = this.editRow();
|
|
179
|
+
if (!current)
|
|
180
|
+
return;
|
|
181
|
+
if (col?.type === 'boolean') {
|
|
182
|
+
// MatCheckboxChange
|
|
183
|
+
const checked = event.checked;
|
|
184
|
+
this.editRow.set({ ...current, [key]: checked });
|
|
185
|
+
}
|
|
186
|
+
else {
|
|
187
|
+
const target = event.target;
|
|
188
|
+
if (target && typeof target.value === 'string') {
|
|
189
|
+
this.editRow.set({ ...current, [key]: target.value });
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
saveEdit(row) {
|
|
194
|
+
const id = row['id'];
|
|
195
|
+
if (typeof id !== 'number' || !this.editRow())
|
|
196
|
+
return;
|
|
197
|
+
const visibleColumns = this.columns.filter(col => col.visible !== false);
|
|
198
|
+
const converted = visibleColumns.reduce((acc, col) => {
|
|
199
|
+
acc[col.accessor] = this.convertCellValue(this.editRow()[col.accessor], col);
|
|
200
|
+
return acc;
|
|
201
|
+
}, {});
|
|
202
|
+
// Type-safe update method detection
|
|
203
|
+
const updateFn = this.service.update;
|
|
204
|
+
if (updateFn) {
|
|
205
|
+
updateFn(id, converted).pipe(take(1)).subscribe({
|
|
206
|
+
next: () => {
|
|
207
|
+
this.cancelEdit();
|
|
208
|
+
this.refresh();
|
|
209
|
+
},
|
|
210
|
+
error: (err) => {
|
|
211
|
+
const message = (typeof err === 'object' && err && 'message' in err) ? err.message : undefined;
|
|
212
|
+
this.errorEvent.emit(message || 'Error al editar');
|
|
213
|
+
}
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).
|
|
219
|
+
*/
|
|
220
|
+
trackById(index, row) {
|
|
221
|
+
const id = row && typeof row === 'object' && 'id' in row ? row['id'] : undefined;
|
|
222
|
+
return (typeof id === 'string' || typeof id === 'number') ? id : index;
|
|
223
|
+
}
|
|
224
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: C80TableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
225
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: C80TableComponent, isStandalone: true, selector: "c80-table", inputs: { service: "service", columns: "columns" }, outputs: { errorEvent: "errorEvent" }, ngImport: i0, template: "<div class=\"table-responsive\">\n <table class=\"table table-bordered table-hover align-middle\">\n <thead class=\"thead table-light\">\n <tr>\n @for (col of columns; track col) {\n @if (col.visible !== false) {\n @if (col.type === 'boolean') {\n <th class=\"text-center\">{{ col.label }}</th>\n }\n @else {\n <th>{{ col.label }}</th>\n }\n }\n }\n <th class=\"table-actions-header\">\n <span>Actions</span>\n <c80-icon button icon=\"add\" [disabled]=\"creating()\" title=\"Agregar\" class=\"add-action-btn\"\n (iconClick)=\"startCreate()\"></c80-icon>\n </th>\n </tr>\n </thead>\n <tbody>\n @for (row of data(); track trackById(i, row); let i = $index) {\n <tr>\n @for (col of columns; track col) {\n @if (col.visible !== false) {\n @if (col.type === 'boolean') {\n <td class=\"text-center\">\n @if (editing() === row['id']) {\n <mat-checkbox [checked]=\"!!editRow()?.[col.accessor]\" (change)=\"onEditInput($event, col.accessor, col)\"\n [aria-label]=\"col.label\"></mat-checkbox>\n }\n @else {\n @if (row[col.accessor] === true) {\n <c80-icon icon=\"check\" [size]=\"1\"></c80-icon>\n <br />\n }\n @else if (row[col.accessor] === false) {\n <c80-icon icon=\"cancel\"></c80-icon>\n <br />\n }\n }\n </td>\n }\n @else {\n <td>\n @if (editing() === row['id']) {\n <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n (input)=\"onEditInput($event, col.accessor, col)\" />\n }\n @else {\n {{ row[col.accessor] }}\n }\n </td>\n }\n }\n }\n <td class=\"text-center\">\n @if (editing() === row['id']) {\n <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveEdit(row)\"></c80-icon>\n <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelEdit()\"></c80-icon>\n }\n @else {\n <c80-icon button icon=\"edit\" title=\"Editar\" (iconClick)=\"onEdit(row)\"></c80-icon>\n <c80-icon button icon=\"delete\" color=\"warn\" title=\"Borrar\" (iconClick)=\"onDelete(row)\"></c80-icon>\n }\n </td>\n </tr>\n }\n @if (creating()) {\n <tr>\n @for (col of columns; track col) {\n @if (col.visible !== false) {\n @if (col.type === 'boolean') {\n <td class=\"text-center\">\n <mat-checkbox [checked]=\"!!newRow()?.[col.accessor]\" (change)=\"onInput($event, col.accessor, col)\"\n [aria-label]=\"col.label\"></mat-checkbox>\n </td>\n } @else {\n <td>\n <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n (input)=\"onInput($event, col.accessor, col)\" />\n </td>\n }\n }\n }\n <td class=\"text-center\">\n <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveCreate()\"></c80-icon>\n <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelCreate()\"></c80-icon>\n </td>\n </tr>\n }\n </tbody>\n </table>\n @if (data().length === 0) {\n <div class=\"text-center text-muted py-3\">\n No hay datos para mostrar.\n </div>\n }\n</div>", styles: ["@charset \"UTF-8\";.table-responsive{width:100%;overflow-x:auto}.table-responsive .table{min-width:0px}.table-responsive .table .thead .table-actions-header{text-align:center;vertical-align:middle;display:flex;justify-content:center;align-items:center;gap:.5rem}.table-responsive .table .thead .table-actions-header .add-action-btn{margin-bottom:2px}.table-responsive .table .thead th{height:35px!important;min-height:35px!important;max-height:35px!important;vertical-align:middle!important;padding:.2rem .6rem!important}.table-responsive .table tbody td{height:35px!important;min-height:35px!important;max-height:35px!important;vertical-align:middle!important;padding:.2rem .8rem!important}.table-responsive .table tbody tr{height:35px!important;min-height:35px!important;max-height:35px!important;cursor:pointer}.table-responsive .table tbody tr:hover{background-color:#f5f5f5}.table-responsive .table tbody input{border:1px solid rgba(34,0,255,.37);height:35px!important}\n"], dependencies: [{ kind: "ngmodule", type: MatCheckboxModule }, { kind: "component", type: i1.MatCheckbox, selector: "mat-checkbox", inputs: ["aria-label", "aria-labelledby", "aria-describedby", "id", "required", "labelPosition", "name", "value", "disableRipple", "tabIndex", "color", "disabledInteractive", "checked", "disabled", "indeterminate"], outputs: ["change", "indeterminateChange"], exportAs: ["matCheckbox"] }, { kind: "component", type: C80IconComponent, selector: "c80-icon", inputs: ["icon", "color", "disabled", "size", "button", "type"], outputs: ["iconClick"] }] });
|
|
226
|
+
}
|
|
227
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: C80TableComponent, decorators: [{
|
|
228
|
+
type: Component,
|
|
229
|
+
args: [{ selector: 'c80-table', standalone: true, imports: [MatCheckboxModule, C80IconComponent], template: "<div class=\"table-responsive\">\n <table class=\"table table-bordered table-hover align-middle\">\n <thead class=\"thead table-light\">\n <tr>\n @for (col of columns; track col) {\n @if (col.visible !== false) {\n @if (col.type === 'boolean') {\n <th class=\"text-center\">{{ col.label }}</th>\n }\n @else {\n <th>{{ col.label }}</th>\n }\n }\n }\n <th class=\"table-actions-header\">\n <span>Actions</span>\n <c80-icon button icon=\"add\" [disabled]=\"creating()\" title=\"Agregar\" class=\"add-action-btn\"\n (iconClick)=\"startCreate()\"></c80-icon>\n </th>\n </tr>\n </thead>\n <tbody>\n @for (row of data(); track trackById(i, row); let i = $index) {\n <tr>\n @for (col of columns; track col) {\n @if (col.visible !== false) {\n @if (col.type === 'boolean') {\n <td class=\"text-center\">\n @if (editing() === row['id']) {\n <mat-checkbox [checked]=\"!!editRow()?.[col.accessor]\" (change)=\"onEditInput($event, col.accessor, col)\"\n [aria-label]=\"col.label\"></mat-checkbox>\n }\n @else {\n @if (row[col.accessor] === true) {\n <c80-icon icon=\"check\" [size]=\"1\"></c80-icon>\n <br />\n }\n @else if (row[col.accessor] === false) {\n <c80-icon icon=\"cancel\"></c80-icon>\n <br />\n }\n }\n </td>\n }\n @else {\n <td>\n @if (editing() === row['id']) {\n <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n (input)=\"onEditInput($event, col.accessor, col)\" />\n }\n @else {\n {{ row[col.accessor] }}\n }\n </td>\n }\n }\n }\n <td class=\"text-center\">\n @if (editing() === row['id']) {\n <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveEdit(row)\"></c80-icon>\n <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelEdit()\"></c80-icon>\n }\n @else {\n <c80-icon button icon=\"edit\" title=\"Editar\" (iconClick)=\"onEdit(row)\"></c80-icon>\n <c80-icon button icon=\"delete\" color=\"warn\" title=\"Borrar\" (iconClick)=\"onDelete(row)\"></c80-icon>\n }\n </td>\n </tr>\n }\n @if (creating()) {\n <tr>\n @for (col of columns; track col) {\n @if (col.visible !== false) {\n @if (col.type === 'boolean') {\n <td class=\"text-center\">\n <mat-checkbox [checked]=\"!!newRow()?.[col.accessor]\" (change)=\"onInput($event, col.accessor, col)\"\n [aria-label]=\"col.label\"></mat-checkbox>\n </td>\n } @else {\n <td>\n <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n (input)=\"onInput($event, col.accessor, col)\" />\n </td>\n }\n }\n }\n <td class=\"text-center\">\n <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveCreate()\"></c80-icon>\n <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelCreate()\"></c80-icon>\n </td>\n </tr>\n }\n </tbody>\n </table>\n @if (data().length === 0) {\n <div class=\"text-center text-muted py-3\">\n No hay datos para mostrar.\n </div>\n }\n</div>", styles: ["@charset \"UTF-8\";.table-responsive{width:100%;overflow-x:auto}.table-responsive .table{min-width:0px}.table-responsive .table .thead .table-actions-header{text-align:center;vertical-align:middle;display:flex;justify-content:center;align-items:center;gap:.5rem}.table-responsive .table .thead .table-actions-header .add-action-btn{margin-bottom:2px}.table-responsive .table .thead th{height:35px!important;min-height:35px!important;max-height:35px!important;vertical-align:middle!important;padding:.2rem .6rem!important}.table-responsive .table tbody td{height:35px!important;min-height:35px!important;max-height:35px!important;vertical-align:middle!important;padding:.2rem .8rem!important}.table-responsive .table tbody tr{height:35px!important;min-height:35px!important;max-height:35px!important;cursor:pointer}.table-responsive .table tbody tr:hover{background-color:#f5f5f5}.table-responsive .table tbody input{border:1px solid rgba(34,0,255,.37);height:35px!important}\n"] }]
|
|
230
|
+
}], propDecorators: { service: [{
|
|
231
|
+
type: Input
|
|
232
|
+
}], columns: [{
|
|
233
|
+
type: Input
|
|
234
|
+
}], errorEvent: [{
|
|
235
|
+
type: Output
|
|
236
|
+
}] } });
|
|
237
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table.component.ts","../../../../../libs/ui/src/lib/table/table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit, Output, signal, EventEmitter } from '@angular/core';\r\nimport { Observable, take } from 'rxjs';\r\nimport { MatCheckboxModule } from '@angular/material/checkbox';\r\nimport type { MatCheckboxChange } from '@angular/material/checkbox';\r\nimport { C80IconComponent } from '../icon/icon.component';\r\n\r\n\r\nexport interface C80TableService<T> {\r\n  get: () => Observable<T[]>;\r\n  create: (row: T) => Observable<T>;\r\n  delete: (id: number) => Observable<void>;\r\n  update: (id: number, row: T) => Observable<T>;\r\n}\r\n\r\nexport interface C80TableColDef {\r\n  accessor: string;\r\n  label: string;\r\n  visible?: boolean; // Si no se especifica, se asume true\r\n  type?: 'string' | 'number' | 'boolean'; // Tipo de dato para la columna\r\n}\r\n\r\n@Component({\r\n  selector: 'c80-table',\r\n  standalone: true,\r\n  imports: [MatCheckboxModule, C80IconComponent],\r\n  templateUrl: './table.component.html',\r\n  styleUrl: './table.component.scss',\r\n})\r\nexport class C80TableComponent<T extends Record<string, unknown>> implements OnInit {\r\n  @Input() service!: C80TableService<T>;\r\n  @Input() columns: C80TableColDef[] = [];\r\n  @Output() errorEvent = new EventEmitter<string>();\r\n\r\n  readonly data = signal<T[]>([]);\r\n  readonly keys = signal<string[]>([]);\r\n  creating = signal(false);\r\n  newRow = signal<Partial<T> | null>(null);\r\n  readonly editing = signal<number | null>(null); // id of row being edited\r\n  readonly editRow = signal<Partial<T> | null>(null);\r\n\r\n  ngOnInit() {\r\n    if (!this.service) return;\r\n    this.refresh();\r\n  }\r\n\r\n  onInput(event: Event | MatCheckboxChange, key: string, col?: C80TableColDef) {\r\n    if (col?.type === 'boolean') {\r\n      // MatCheckboxChange\r\n      const checked = (event as MatCheckboxChange).checked;\r\n      this.updateNewRow(key, checked);\r\n    } else {\r\n      const target = (event as Event).target;\r\n      if (target && typeof (target as HTMLInputElement).value === 'string') {\r\n        this.updateNewRow(key, (target as HTMLInputElement).value);\r\n      }\r\n    }\r\n  }\r\n\r\n  onDelete(row: T) {\r\n    if (!this.service.delete) return;\r\n    const id = row['id'];\r\n    if (typeof id !== 'number') {\r\n      this.errorEvent.emit('No se puede borrar: id inválido');\r\n      return;\r\n    }\r\n    if (confirm('¿Está seguro de que desea borrar este elemento?')) {\r\n      this.service.delete(id).pipe(take(1)).subscribe({\r\n        next: () => this.refresh(),\r\n        error: (err) => this.errorEvent.emit(err?.message || 'Error al borrar')\r\n      });\r\n    }\r\n  }\r\n\r\n  refresh() {\r\n    this.service.get().pipe(take(1)).subscribe((items) => {\r\n      this.data.update(() => items);\r\n      // Solo mostrar las columnas visibles\r\n      const visibleColumns = this.columns.filter(col => col?.visible !== false);\r\n      this.keys.update(() => visibleColumns.map(col => col.accessor));\r\n    });\r\n  }\r\n\r\n  startCreate() {\r\n    this.creating.set(true);\r\n    // Inicializa newRow solo con los accessors de columnas visibles\r\n    const row: Partial<T> = {};\r\n    this.columns.filter(col => col.visible !== false).forEach(col => {\r\n      (row as Record<string, unknown>)[col.accessor] = '';\r\n    });\r\n    this.newRow.set(row);\r\n  }\r\n\r\n  cancelCreate() {\r\n    this.creating.set(false);\r\n    this.newRow.set(null);\r\n  }\r\n\r\n  updateNewRow(key: string, value: unknown) {\r\n    const current = this.newRow();\r\n    if (!current) return;\r\n    this.newRow.set({ ...current, [key]: value });\r\n  }\r\n\r\n  saveCreate() {\r\n    const row = this.newRow();\r\n    if (!row) return;\r\n\r\n    const visibleColumns = this.columns.filter(col => col.visible !== false);\r\n    const converted = visibleColumns.reduce((acc, col) => {\r\n      acc[col.accessor] = this.convertCellValue(row[col.accessor], col);\r\n      return acc;\r\n    }, {} as Record<string, unknown>) as Partial<T>;\r\n\r\n    if (this.service.create) {\r\n      (this.service.create as (row: Partial<T>) => Observable<T>)(converted).pipe(take(1)).subscribe({\r\n        next: () => {\r\n          this.cancelCreate();\r\n          this.refresh();\r\n        },\r\n        error: (err) => {\r\n          this.errorEvent.emit(err?.message || 'Error al crear');\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Converts a cell value based on column type or sample data.\r\n   * Handles stringification of objects for string columns.\r\n   */\r\n  /**\r\n   * Converts a cell value based on column type or sample data.\r\n   * Delegates to type-specific helpers for clarity and maintainability.\r\n   */\r\n  private convertCellValue(value: unknown, col: C80TableColDef): unknown {\r\n    if (col.type === 'boolean') return this.toBoolean(value);\r\n    if (col.type === 'number') return this.toNumber(value);\r\n    if (col.type === 'string') return this.toStringValue(value);\r\n\r\n    // Fallback: use sample data if available\r\n    const sample = this.data()[0];\r\n    if (sample && col.accessor in sample) {\r\n      const sampleValue = sample[col.accessor];\r\n      if (typeof sampleValue === 'boolean') return this.toBoolean(value);\r\n      if (typeof sampleValue === 'number') return this.toNumber(value);\r\n      if (typeof sampleValue === 'string') return this.toStringValue(value);\r\n    }\r\n    return value;\r\n  }\r\n\r\n  /** Converts value to boolean using best practices. */\r\n  private toBoolean(value: unknown): boolean {\r\n    if (typeof value === 'boolean') return value;\r\n    if (typeof value === 'string') return value.trim().toLowerCase() === 'true' || value.trim() === '1';\r\n    if (typeof value === 'number') return value === 1;\r\n    return false;\r\n  }\r\n\r\n  /** Converts value to number using best practices. */\r\n  private toNumber(value: unknown): number | undefined {\r\n    if (typeof value === 'number') return value;\r\n    if (typeof value === 'string') return value.trim() === '' ? undefined : Number(value);\r\n    if (typeof value === 'boolean') return value ? 1 : 0;\r\n    return undefined;\r\n  }\r\n\r\n  /** Converts value to string using best practices, always stringifies objects. */\r\n  private toStringValue(value: unknown): string {\r\n    if (value == null) return '';\r\n    if (typeof value === 'string') return value;\r\n    if (typeof value === 'number' || typeof value === 'boolean') return String(value);\r\n    if (typeof value === 'object') {\r\n      try {\r\n        return JSON.stringify(value);\r\n      } catch {\r\n        return '[object Object]';\r\n      }\r\n    }\r\n    // For functions, symbols, undefined, etc., return empty string\r\n    return '';\r\n  }\r\n\r\n  onEdit(row: T) {\r\n    this.editing.set(row['id'] as number);\r\n    const edit: Partial<T> = {};\r\n    this.columns.filter(col => col.visible !== false).forEach(col => {\r\n      (edit as Record<string, unknown>)[col.accessor] = row[col.accessor];\r\n    });\r\n    this.editRow.set(edit);\r\n  }\r\n\r\n  cancelEdit() {\r\n    this.editing.set(null);\r\n    this.editRow.set(null);\r\n  }\r\n\r\n  onEditInput(event: Event | MatCheckboxChange, key: string, col?: C80TableColDef) {\r\n    const current = this.editRow();\r\n    if (!current) return;\r\n    if (col?.type === 'boolean') {\r\n      // MatCheckboxChange\r\n      const checked = (event as MatCheckboxChange).checked;\r\n      this.editRow.set({ ...current, [key]: checked });\r\n    } else {\r\n      const target = (event as Event).target;\r\n      if (target && typeof (target as HTMLInputElement).value === 'string') {\r\n        this.editRow.set({ ...current, [key]: (target as HTMLInputElement).value });\r\n      }\r\n    }\r\n  }\r\n\r\n  saveEdit(row: T) {\r\n    const id = row['id'];\r\n    if (typeof id !== 'number' || !this.editRow()) return;\r\n    const visibleColumns = this.columns.filter(col => col.visible !== false);\r\n    const converted = visibleColumns.reduce((acc, col) => {\r\n      acc[col.accessor] = this.convertCellValue(this.editRow()![col.accessor], col);\r\n      return acc;\r\n    }, {} as Record<string, unknown>) as Partial<T>;\r\n    // Type-safe update method detection\r\n    const updateFn = (this.service as { update?: (id: number, row: T) => Observable<T> }).update;\r\n    if (updateFn) {\r\n      updateFn(id, converted as T).pipe(take(1)).subscribe({\r\n        next: () => {\r\n          this.cancelEdit();\r\n          this.refresh();\r\n        },\r\n        error: (err: unknown) => {\r\n          const message = (typeof err === 'object' && err && 'message' in err) ? (err as { message?: string }).message : undefined;\r\n          this.errorEvent.emit(message || 'Error al editar');\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  /**\r\n   * TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).\r\n   */\r\n  trackById(index: number, row: T): number | string {\r\n    const id = row && typeof row === 'object' && 'id' in row ? (row as Record<string, unknown>)['id'] : undefined;\r\n    return (typeof id === 'string' || typeof id === 'number') ? id : index;\r\n  }\r\n}\r\n","<div class=\"table-responsive\">\n  <table class=\"table table-bordered table-hover align-middle\">\n    <thead class=\"thead table-light\">\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <th class=\"text-center\">{{ col.label }}</th>\n        }\n        @else {\n        <th>{{ col.label }}</th>\n        }\n        }\n        }\n        <th class=\"table-actions-header\">\n          <span>Actions</span>\n          <c80-icon button icon=\"add\" [disabled]=\"creating()\" title=\"Agregar\" class=\"add-action-btn\"\n            (iconClick)=\"startCreate()\"></c80-icon>\n        </th>\n      </tr>\n    </thead>\n    <tbody>\n      @for (row of data(); track trackById(i, row); let i = $index) {\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center\">\n          @if (editing() === row['id']) {\n          <mat-checkbox [checked]=\"!!editRow()?.[col.accessor]\" (change)=\"onEditInput($event, col.accessor, col)\"\n            [aria-label]=\"col.label\"></mat-checkbox>\n          }\n          @else {\n          @if (row[col.accessor] === true) {\n          <c80-icon icon=\"check\" [size]=\"1\"></c80-icon>\n          <br />\n          }\n          @else if (row[col.accessor] === false) {\n          <c80-icon icon=\"cancel\"></c80-icon>\n          <br />\n          }\n          }\n        </td>\n        }\n        @else {\n        <td>\n          @if (editing() === row['id']) {\n          <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n            [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n            (input)=\"onEditInput($event, col.accessor, col)\" />\n          }\n          @else {\n          {{ row[col.accessor] }}\n          }\n        </td>\n        }\n        }\n        }\n        <td class=\"text-center\">\n          @if (editing() === row['id']) {\n          <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveEdit(row)\"></c80-icon>\n          <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelEdit()\"></c80-icon>\n          }\n          @else {\n          <c80-icon button icon=\"edit\" title=\"Editar\" (iconClick)=\"onEdit(row)\"></c80-icon>\n          <c80-icon button icon=\"delete\" color=\"warn\" title=\"Borrar\" (iconClick)=\"onDelete(row)\"></c80-icon>\n          }\n        </td>\n      </tr>\n      }\n      @if (creating()) {\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center\">\n          <mat-checkbox [checked]=\"!!newRow()?.[col.accessor]\" (change)=\"onInput($event, col.accessor, col)\"\n            [aria-label]=\"col.label\"></mat-checkbox>\n        </td>\n        } @else {\n        <td>\n          <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n            [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n            (input)=\"onInput($event, col.accessor, col)\" />\n        </td>\n        }\n        }\n        }\n        <td class=\"text-center\">\n          <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveCreate()\"></c80-icon>\n          <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelCreate()\"></c80-icon>\n        </td>\n      </tr>\n      }\n    </tbody>\n  </table>\n  @if (data().length === 0) {\n  <div class=\"text-center text-muted py-3\">\n    No hay datos para mostrar.\n  </div>\n  }\n</div>"]}
|
package/index.d.ts
CHANGED
|
@@ -1 +1,2 @@
|
|
|
1
|
-
export * from './lib/
|
|
1
|
+
export * from './lib/table/table.component';
|
|
2
|
+
export * from './lib/icon/icon.component';
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { EventEmitter, TemplateRef } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
export type Icon = 'check' | 'cancel' | 'edit' | 'delete' | 'add';
|
|
4
|
+
export type Color = 'primary' | 'secondary' | 'warn';
|
|
5
|
+
export declare class C80IconComponent {
|
|
6
|
+
/** Tamaño base del icono (en px) */
|
|
7
|
+
get iconSize(): number;
|
|
8
|
+
icon: Icon;
|
|
9
|
+
color: Color;
|
|
10
|
+
disabled?: boolean;
|
|
11
|
+
size: number;
|
|
12
|
+
/** Si es true, renderiza como <button> nativo, si es false solo como icono */
|
|
13
|
+
private _button;
|
|
14
|
+
set button(val: boolean | string | null | undefined);
|
|
15
|
+
get button(): boolean;
|
|
16
|
+
/** Tipo de botón nativo (solo si button=true) */
|
|
17
|
+
type: 'button' | 'submit' | 'reset';
|
|
18
|
+
/** Output para click (solo si button=true) */
|
|
19
|
+
iconClick: EventEmitter<Event>;
|
|
20
|
+
checkIconTpl: TemplateRef<unknown>;
|
|
21
|
+
cancelIconTpl: TemplateRef<unknown>;
|
|
22
|
+
editIconTpl: TemplateRef<unknown>;
|
|
23
|
+
deleteIconTpl: TemplateRef<unknown>;
|
|
24
|
+
addIconTpl: TemplateRef<unknown>;
|
|
25
|
+
defaultIconTpl: TemplateRef<unknown>;
|
|
26
|
+
getIconTemplate(): TemplateRef<unknown>;
|
|
27
|
+
get iconColor(): string;
|
|
28
|
+
onButtonClick(event: Event): void;
|
|
29
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<C80IconComponent, never>;
|
|
30
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<C80IconComponent, "c80-icon", never, { "icon": { "alias": "icon"; "required": false; }; "color": { "alias": "color"; "required": false; }; "disabled": { "alias": "disabled"; "required": false; }; "size": { "alias": "size"; "required": false; }; "button": { "alias": "button"; "required": false; }; "type": { "alias": "type"; "required": false; }; }, { "iconClick": "iconClick"; }, never, never, true, never>;
|
|
31
|
+
}
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import { OnInit, EventEmitter } from '@angular/core';
|
|
2
|
+
import { Observable } from 'rxjs';
|
|
3
|
+
import type { MatCheckboxChange } from '@angular/material/checkbox';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
export interface C80TableService<T> {
|
|
6
|
+
get: () => Observable<T[]>;
|
|
7
|
+
create: (row: T) => Observable<T>;
|
|
8
|
+
delete: (id: number) => Observable<void>;
|
|
9
|
+
update: (id: number, row: T) => Observable<T>;
|
|
10
|
+
}
|
|
11
|
+
export interface C80TableColDef {
|
|
12
|
+
accessor: string;
|
|
13
|
+
label: string;
|
|
14
|
+
visible?: boolean;
|
|
15
|
+
type?: 'string' | 'number' | 'boolean';
|
|
16
|
+
}
|
|
17
|
+
export declare class C80TableComponent<T extends Record<string, unknown>> implements OnInit {
|
|
18
|
+
service: C80TableService<T>;
|
|
19
|
+
columns: C80TableColDef[];
|
|
20
|
+
errorEvent: EventEmitter<string>;
|
|
21
|
+
readonly data: import("@angular/core").WritableSignal<T[]>;
|
|
22
|
+
readonly keys: import("@angular/core").WritableSignal<string[]>;
|
|
23
|
+
creating: import("@angular/core").WritableSignal<boolean>;
|
|
24
|
+
newRow: import("@angular/core").WritableSignal<Partial<T> | null>;
|
|
25
|
+
readonly editing: import("@angular/core").WritableSignal<number | null>;
|
|
26
|
+
readonly editRow: import("@angular/core").WritableSignal<Partial<T> | null>;
|
|
27
|
+
ngOnInit(): void;
|
|
28
|
+
onInput(event: Event | MatCheckboxChange, key: string, col?: C80TableColDef): void;
|
|
29
|
+
onDelete(row: T): void;
|
|
30
|
+
refresh(): void;
|
|
31
|
+
startCreate(): void;
|
|
32
|
+
cancelCreate(): void;
|
|
33
|
+
updateNewRow(key: string, value: unknown): void;
|
|
34
|
+
saveCreate(): void;
|
|
35
|
+
/**
|
|
36
|
+
* Converts a cell value based on column type or sample data.
|
|
37
|
+
* Handles stringification of objects for string columns.
|
|
38
|
+
*/
|
|
39
|
+
/**
|
|
40
|
+
* Converts a cell value based on column type or sample data.
|
|
41
|
+
* Delegates to type-specific helpers for clarity and maintainability.
|
|
42
|
+
*/
|
|
43
|
+
private convertCellValue;
|
|
44
|
+
/** Converts value to boolean using best practices. */
|
|
45
|
+
private toBoolean;
|
|
46
|
+
/** Converts value to number using best practices. */
|
|
47
|
+
private toNumber;
|
|
48
|
+
/** Converts value to string using best practices, always stringifies objects. */
|
|
49
|
+
private toStringValue;
|
|
50
|
+
onEdit(row: T): void;
|
|
51
|
+
cancelEdit(): void;
|
|
52
|
+
onEditInput(event: Event | MatCheckboxChange, key: string, col?: C80TableColDef): void;
|
|
53
|
+
saveEdit(row: T): void;
|
|
54
|
+
/**
|
|
55
|
+
* TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).
|
|
56
|
+
*/
|
|
57
|
+
trackById(index: number, row: T): number | string;
|
|
58
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<C80TableComponent<any>, never>;
|
|
59
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<C80TableComponent<any>, "c80-table", never, { "service": { "alias": "service"; "required": false; }; "columns": { "alias": "columns"; "required": false; }; }, { "errorEvent": "errorEvent"; }, never, never, true, never>;
|
|
60
|
+
}
|
package/package.json
CHANGED
|
@@ -1,9 +1,11 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@c80/ui",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "1.0.1",
|
|
4
4
|
"peerDependencies": {
|
|
5
|
-
"@angular/
|
|
6
|
-
"
|
|
5
|
+
"@angular/core": "^18.2.0",
|
|
6
|
+
"rxjs": "~7.8.0",
|
|
7
|
+
"@angular/material": "^18.2.14",
|
|
8
|
+
"@angular/common": "~18.2.0"
|
|
7
9
|
},
|
|
8
10
|
"sideEffects": false,
|
|
9
11
|
"module": "esm2022/c80-ui.mjs",
|
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component } from '@angular/core';
|
|
2
|
-
import { CommonModule } from '@angular/common';
|
|
3
|
-
import * as i0 from "@angular/core";
|
|
4
|
-
export class UiComponent {
|
|
5
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UiComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
6
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: UiComponent, isStandalone: true, selector: "lib-ui", ngImport: i0, template: "<p>Ui works!</p>\n", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
7
|
-
}
|
|
8
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: UiComponent, decorators: [{
|
|
9
|
-
type: Component,
|
|
10
|
-
args: [{ selector: 'lib-ui', standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, template: "<p>Ui works!</p>\n" }]
|
|
11
|
-
}] });
|
|
12
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidWkuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vbGlicy91aS9zcmMvbGliL3VpL3VpLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvc3JjL2xpYi91aS91aS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ25FLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQzs7QUFVL0MsTUFBTSxPQUFPLFdBQVc7d0dBQVgsV0FBVzs0RkFBWCxXQUFXLGtFQ1h4QixvQkFDQSx5RERLWSxZQUFZOzs0RkFLWCxXQUFXO2tCQVJ2QixTQUFTOytCQUNFLFFBQVEsY0FDTixJQUFJLFdBQ1AsQ0FBQyxZQUFZLENBQUMsbUJBR04sdUJBQXVCLENBQUMsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2xpYi11aScsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdLFxuICB0ZW1wbGF0ZVVybDogJy4vdWkuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybDogJy4vdWkuY29tcG9uZW50LmNzcycsXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxufSlcbmV4cG9ydCBjbGFzcyBVaUNvbXBvbmVudCB7fVxuIiwiPHA+VWkgd29ya3MhPC9wPlxuIl19
|
package/lib/ui/ui.component.d.ts
DELETED