pdm-ui-kit 0.1.49 → 0.2.0

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.
Files changed (52) hide show
  1. package/README.md +189 -2
  2. package/esm2020/lib/components/alert-dialog/alert-dialog.component.mjs +25 -7
  3. package/esm2020/lib/components/breadcrumb/breadcrumb.component.mjs +37 -4
  4. package/esm2020/lib/components/calendar/calendar.component.mjs +3 -3
  5. package/esm2020/lib/components/card/card.component.mjs +36 -53
  6. package/esm2020/lib/components/command/command.component.mjs +3 -3
  7. package/esm2020/lib/components/context-menu/context-menu.component.mjs +16 -8
  8. package/esm2020/lib/components/data-table/data-table.component.mjs +214 -16
  9. package/esm2020/lib/components/dialog/dialog.component.mjs +133 -17
  10. package/esm2020/lib/components/draggable-table/draggable-table.component.mjs +300 -0
  11. package/esm2020/lib/components/drawer/drawer.component.mjs +138 -10
  12. package/esm2020/lib/components/dropdown-menu/dropdown-menu.component.mjs +6 -3
  13. package/esm2020/lib/components/hover-card/hover-card.component.mjs +3 -3
  14. package/esm2020/lib/components/menubar/menubar.component.mjs +38 -7
  15. package/esm2020/lib/components/navigation-menu/navigation-menu.component.mjs +25 -3
  16. package/esm2020/lib/components/pagination/pagination.component.mjs +3 -3
  17. package/esm2020/lib/components/popover/popover.component.mjs +19 -11
  18. package/esm2020/lib/components/select/select.component.mjs +8 -4
  19. package/esm2020/lib/components/sheet/sheet.component.mjs +88 -11
  20. package/esm2020/lib/components/sidebar/sidebar.component.mjs +52 -5
  21. package/esm2020/lib/components/table/table.component.mjs +152 -188
  22. package/esm2020/lib/components/tabs/tabs.component.mjs +3 -3
  23. package/esm2020/lib/components/tooltip/tooltip.component.mjs +3 -3
  24. package/esm2020/lib/overlay/pdm-outside-click.directive.mjs +86 -0
  25. package/esm2020/lib/pdm-ui-kit.module.mjs +9 -1
  26. package/esm2020/lib/utils/responsive.mjs +143 -0
  27. package/esm2020/lib/utils/z-index.mjs +93 -0
  28. package/esm2020/public-api.mjs +5 -1
  29. package/fesm2015/pdm-ui-kit.mjs +1625 -370
  30. package/fesm2015/pdm-ui-kit.mjs.map +1 -1
  31. package/fesm2020/pdm-ui-kit.mjs +1620 -367
  32. package/fesm2020/pdm-ui-kit.mjs.map +1 -1
  33. package/lib/components/alert-dialog/alert-dialog.component.d.ts +9 -1
  34. package/lib/components/breadcrumb/breadcrumb.component.d.ts +23 -1
  35. package/lib/components/card/card.component.d.ts +32 -19
  36. package/lib/components/context-menu/context-menu.component.d.ts +6 -3
  37. package/lib/components/data-table/data-table.component.d.ts +172 -14
  38. package/lib/components/dialog/dialog.component.d.ts +35 -1
  39. package/lib/components/draggable-table/draggable-table.component.d.ts +74 -0
  40. package/lib/components/drawer/drawer.component.d.ts +67 -3
  41. package/lib/components/menubar/menubar.component.d.ts +10 -2
  42. package/lib/components/navigation-menu/navigation-menu.component.d.ts +22 -1
  43. package/lib/components/popover/popover.component.d.ts +6 -3
  44. package/lib/components/sheet/sheet.component.d.ts +34 -1
  45. package/lib/components/sidebar/sidebar.component.d.ts +39 -1
  46. package/lib/components/table/table.component.d.ts +46 -25
  47. package/lib/overlay/pdm-outside-click.directive.d.ts +40 -0
  48. package/lib/pdm-ui-kit.module.d.ts +42 -40
  49. package/lib/utils/responsive.d.ts +107 -0
  50. package/lib/utils/z-index.d.ts +73 -0
  51. package/package.json +5 -3
  52. package/public-api.d.ts +4 -0
@@ -27,6 +27,15 @@ export class PdmContextMenuComponent {
27
27
  this.x = 0;
28
28
  this.y = 0;
29
29
  }
30
+ ngOnInit() {
31
+ this.boundPointerDown = (event) => this.onDocumentPointerDown(event);
32
+ document.addEventListener('pointerdown', this.boundPointerDown, { capture: true });
33
+ }
34
+ ngOnDestroy() {
35
+ if (this.boundPointerDown) {
36
+ document.removeEventListener('pointerdown', this.boundPointerDown, { capture: true });
37
+ }
38
+ }
30
39
  onContextMenu(event) {
31
40
  event.preventDefault();
32
41
  this.x = event.clientX;
@@ -40,9 +49,11 @@ export class PdmContextMenuComponent {
40
49
  this.open = false;
41
50
  }
42
51
  onEsc() {
43
- this.open = false;
52
+ if (this.open) {
53
+ this.open = false;
54
+ }
44
55
  }
45
- onDocumentClick(event) {
56
+ onDocumentPointerDown(event) {
46
57
  if (!this.open)
47
58
  return;
48
59
  const target = event.target;
@@ -52,10 +63,10 @@ export class PdmContextMenuComponent {
52
63
  }
53
64
  }
54
65
  PdmContextMenuComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmContextMenuComponent, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
55
- PdmContextMenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PdmContextMenuComponent, selector: "pdm-context-menu", inputs: { items: "items", className: "className", triggerLabel: "triggerLabel", width: "width", height: "height" }, outputs: { itemSelect: "itemSelect" }, host: { listeners: { "document:keydown.escape": "onEsc()", "document:click": "onDocumentClick($event)" } }, ngImport: i0, template: "<div class=\"relative\" [ngClass]=\"className\" (contextmenu)=\"onContextMenu($event)\">\n <div\n class=\"flex items-center justify-center rounded-md border border-dashed border-border\"\n [style.width.px]=\"width\"\n [style.height.px]=\"height\"\n >\n <span class=\"text-sm font-medium text-foreground\">{{ triggerLabel }}</span>\n </div>\n\n <div\n *ngIf=\"open\"\n class=\"fixed z-50 w-52 rounded-md border border-border bg-popover p-1 text-popover-foreground shadow-md\"\n [style.left.px]=\"x + 4\"\n [style.top.px]=\"y + 2\"\n >\n <div>\n <ng-container *ngFor=\"let item of items\">\n <div *ngIf=\"item.type === 'separator'\" class=\"-mx-1 my-1 h-px bg-muted\"></div>\n\n <div *ngIf=\"item.type === 'label'\" class=\"px-2 py-1.5 text-sm font-semibold text-foreground\">\n {{ item.label }}\n </div>\n\n <button\n *ngIf=\"!item.type || item.type === 'item'\"\n type=\"button\"\n [disabled]=\"item.disabled\"\n class=\"relative flex w-full appearance-none cursor-default select-none items-center rounded-sm border-0 bg-transparent py-1.5 pr-2 text-left text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground focus-visible:bg-accent focus-visible:text-accent-foreground disabled:pointer-events-none disabled:opacity-50\"\n [ngClass]=\"item.inset ? 'pl-8' : 'px-2'\"\n (click)=\"select(item)\"\n >\n <span class=\"mr-2 inline-flex w-4 shrink-0 items-center justify-center text-foreground\">\n <svg *ngIf=\"item.checked\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M5 12.5L9.2 16.7L19 7\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n <span *ngIf=\"item.selectedDot\" class=\"h-2 w-2 rounded-full bg-foreground\"></span>\n </span>\n <span class=\"min-w-0 flex-1 truncate text-foreground\">{{ item.label }}</span>\n <span *ngIf=\"item.shortcut\" class=\"text-xs text-muted-foreground\">{{ item.shortcut }}</span>\n <svg *ngIf=\"item.showChevron\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5 text-muted-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M9 6L15 12L9 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </ng-container>\n </div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
66
+ PdmContextMenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PdmContextMenuComponent, selector: "pdm-context-menu", inputs: { items: "items", className: "className", triggerLabel: "triggerLabel", width: "width", height: "height" }, outputs: { itemSelect: "itemSelect" }, host: { listeners: { "document:keydown.escape": "onEsc()" } }, ngImport: i0, template: "<div class=\"relative\" [ngClass]=\"className\" (contextmenu)=\"onContextMenu($event)\">\n <div\n class=\"flex items-center justify-center rounded-md border border-dashed border-border\"\n [style.width.px]=\"width\"\n [style.height.px]=\"height\"\n >\n <span class=\"text-sm font-medium text-foreground\">{{ triggerLabel }}</span>\n </div>\n\n <div\n *ngIf=\"open\"\n class=\"fixed z-[70] min-w-48 max-w-xs rounded-md border border-border bg-popover p-1 text-popover-foreground shadow-md sm:min-w-52\"\n [style.left.px]=\"x + 4\"\n [style.top.px]=\"y + 2\"\n >\n <div>\n <ng-container *ngFor=\"let item of items\">\n <div *ngIf=\"item.type === 'separator'\" class=\"-mx-1 my-1 h-px bg-muted\"></div>\n\n <div *ngIf=\"item.type === 'label'\" class=\"px-2 py-1.5 text-sm font-semibold text-foreground\">\n {{ item.label }}\n </div>\n\n <button\n *ngIf=\"!item.type || item.type === 'item'\"\n type=\"button\"\n [disabled]=\"item.disabled\"\n class=\"relative flex w-full appearance-none cursor-default select-none items-center rounded-sm border-0 bg-transparent py-1.5 pr-2 text-left text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground focus-visible:bg-accent focus-visible:text-accent-foreground disabled:pointer-events-none disabled:opacity-50\"\n [ngClass]=\"item.inset ? 'pl-8' : 'px-2'\"\n (click)=\"select(item)\"\n >\n <span class=\"mr-2 inline-flex w-4 shrink-0 items-center justify-center text-foreground\">\n <svg *ngIf=\"item.checked\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M5 12.5L9.2 16.7L19 7\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n <span *ngIf=\"item.selectedDot\" class=\"h-2 w-2 rounded-full bg-foreground\"></span>\n </span>\n <span class=\"min-w-0 flex-1 truncate text-foreground\">{{ item.label }}</span>\n <span *ngIf=\"item.shortcut\" class=\"text-xs text-muted-foreground\">{{ item.shortcut }}</span>\n <svg *ngIf=\"item.showChevron\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5 text-muted-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M9 6L15 12L9 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </ng-container>\n </div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
56
67
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmContextMenuComponent, decorators: [{
57
68
  type: Component,
58
- args: [{ selector: 'pdm-context-menu', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"relative\" [ngClass]=\"className\" (contextmenu)=\"onContextMenu($event)\">\n <div\n class=\"flex items-center justify-center rounded-md border border-dashed border-border\"\n [style.width.px]=\"width\"\n [style.height.px]=\"height\"\n >\n <span class=\"text-sm font-medium text-foreground\">{{ triggerLabel }}</span>\n </div>\n\n <div\n *ngIf=\"open\"\n class=\"fixed z-50 w-52 rounded-md border border-border bg-popover p-1 text-popover-foreground shadow-md\"\n [style.left.px]=\"x + 4\"\n [style.top.px]=\"y + 2\"\n >\n <div>\n <ng-container *ngFor=\"let item of items\">\n <div *ngIf=\"item.type === 'separator'\" class=\"-mx-1 my-1 h-px bg-muted\"></div>\n\n <div *ngIf=\"item.type === 'label'\" class=\"px-2 py-1.5 text-sm font-semibold text-foreground\">\n {{ item.label }}\n </div>\n\n <button\n *ngIf=\"!item.type || item.type === 'item'\"\n type=\"button\"\n [disabled]=\"item.disabled\"\n class=\"relative flex w-full appearance-none cursor-default select-none items-center rounded-sm border-0 bg-transparent py-1.5 pr-2 text-left text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground focus-visible:bg-accent focus-visible:text-accent-foreground disabled:pointer-events-none disabled:opacity-50\"\n [ngClass]=\"item.inset ? 'pl-8' : 'px-2'\"\n (click)=\"select(item)\"\n >\n <span class=\"mr-2 inline-flex w-4 shrink-0 items-center justify-center text-foreground\">\n <svg *ngIf=\"item.checked\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M5 12.5L9.2 16.7L19 7\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n <span *ngIf=\"item.selectedDot\" class=\"h-2 w-2 rounded-full bg-foreground\"></span>\n </span>\n <span class=\"min-w-0 flex-1 truncate text-foreground\">{{ item.label }}</span>\n <span *ngIf=\"item.shortcut\" class=\"text-xs text-muted-foreground\">{{ item.shortcut }}</span>\n <svg *ngIf=\"item.showChevron\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5 text-muted-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M9 6L15 12L9 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </ng-container>\n </div>\n </div>\n</div>\n" }]
69
+ args: [{ selector: 'pdm-context-menu', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"relative\" [ngClass]=\"className\" (contextmenu)=\"onContextMenu($event)\">\n <div\n class=\"flex items-center justify-center rounded-md border border-dashed border-border\"\n [style.width.px]=\"width\"\n [style.height.px]=\"height\"\n >\n <span class=\"text-sm font-medium text-foreground\">{{ triggerLabel }}</span>\n </div>\n\n <div\n *ngIf=\"open\"\n class=\"fixed z-[70] min-w-48 max-w-xs rounded-md border border-border bg-popover p-1 text-popover-foreground shadow-md sm:min-w-52\"\n [style.left.px]=\"x + 4\"\n [style.top.px]=\"y + 2\"\n >\n <div>\n <ng-container *ngFor=\"let item of items\">\n <div *ngIf=\"item.type === 'separator'\" class=\"-mx-1 my-1 h-px bg-muted\"></div>\n\n <div *ngIf=\"item.type === 'label'\" class=\"px-2 py-1.5 text-sm font-semibold text-foreground\">\n {{ item.label }}\n </div>\n\n <button\n *ngIf=\"!item.type || item.type === 'item'\"\n type=\"button\"\n [disabled]=\"item.disabled\"\n class=\"relative flex w-full appearance-none cursor-default select-none items-center rounded-sm border-0 bg-transparent py-1.5 pr-2 text-left text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground focus-visible:bg-accent focus-visible:text-accent-foreground disabled:pointer-events-none disabled:opacity-50\"\n [ngClass]=\"item.inset ? 'pl-8' : 'px-2'\"\n (click)=\"select(item)\"\n >\n <span class=\"mr-2 inline-flex w-4 shrink-0 items-center justify-center text-foreground\">\n <svg *ngIf=\"item.checked\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M5 12.5L9.2 16.7L19 7\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n <span *ngIf=\"item.selectedDot\" class=\"h-2 w-2 rounded-full bg-foreground\"></span>\n </span>\n <span class=\"min-w-0 flex-1 truncate text-foreground\">{{ item.label }}</span>\n <span *ngIf=\"item.shortcut\" class=\"text-xs text-muted-foreground\">{{ item.shortcut }}</span>\n <svg *ngIf=\"item.showChevron\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5 text-muted-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M9 6L15 12L9 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </ng-container>\n </div>\n </div>\n</div>\n" }]
59
70
  }], ctorParameters: function () { return [{ type: i0.ElementRef }]; }, propDecorators: { items: [{
60
71
  type: Input
61
72
  }], className: [{
@@ -71,8 +82,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
71
82
  }], onEsc: [{
72
83
  type: HostListener,
73
84
  args: ['document:keydown.escape']
74
- }], onDocumentClick: [{
75
- type: HostListener,
76
- args: ['document:click', ['$event']]
77
85
  }] } });
78
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-menu.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/context-menu/context-menu.component.ts","../../../../../../src/lib/components/context-menu/context-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, HostListener, Input, Output } from '@angular/core';\nimport { PdmMenuItem } from '../dropdown-menu/dropdown-menu.component';\n\nexport interface PdmContextMenuItem extends PdmMenuItem {\n  checked?: boolean;\n  selectedDot?: boolean;\n}\n\n@Component({\n  selector: 'pdm-context-menu',\n  templateUrl: './context-menu.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PdmContextMenuComponent {\n  @Input() items: PdmContextMenuItem[] = [\n    { type: 'item', label: 'Back', value: 'back', inset: true, shortcut: '⌘[' },\n    { type: 'item', label: 'Forward', value: 'forward', inset: true, shortcut: '⌘]', disabled: true },\n    { type: 'item', label: 'Reload', value: 'reload', inset: true, shortcut: '⌘R' },\n    { type: 'item', label: 'More Tools', value: 'more-tools', inset: true, showChevron: true },\n    { type: 'separator' },\n    { type: 'item', label: 'Show Bookmarks Bar', value: 'show-bookmarks', checked: true },\n    { type: 'item', label: 'Show Full URLs', value: 'show-urls', inset: true },\n    { type: 'separator' },\n    { type: 'label', label: 'People' },\n    { type: 'separator' },\n    { type: 'item', label: 'Pedro Duarte', value: 'pedro', selectedDot: true },\n    { type: 'item', label: 'Colm Tuite', value: 'colm', inset: true }\n  ];\n  @Input() className = '';\n  @Input() triggerLabel = 'Right click here';\n  @Input() width = 300;\n  @Input() height = 150;\n  @Output() itemSelect = new EventEmitter<string>();\n\n  open = false;\n  x = 0;\n  y = 0;\n\n  constructor(private readonly elementRef: ElementRef<HTMLElement>) {}\n\n  onContextMenu(event: MouseEvent): void {\n    event.preventDefault();\n    this.x = event.clientX;\n    this.y = event.clientY;\n    this.open = true;\n  }\n\n  select(item: PdmContextMenuItem): void {\n    if (item.disabled || item.type === 'separator' || item.type === 'label' || !item.value) return;\n    this.itemSelect.emit(item.value);\n    this.open = false;\n  }\n\n  @HostListener('document:keydown.escape')\n  onEsc(): void {\n    this.open = false;\n  }\n\n  @HostListener('document:click', ['$event'])\n  onDocumentClick(event: MouseEvent): void {\n    if (!this.open) return;\n    const target = event.target as Node | null;\n    if (target && !this.elementRef.nativeElement.contains(target)) {\n      this.open = false;\n    }\n  }\n}\n","<div class=\"relative\" [ngClass]=\"className\" (contextmenu)=\"onContextMenu($event)\">\n  <div\n    class=\"flex items-center justify-center rounded-md border border-dashed border-border\"\n    [style.width.px]=\"width\"\n    [style.height.px]=\"height\"\n  >\n    <span class=\"text-sm font-medium text-foreground\">{{ triggerLabel }}</span>\n  </div>\n\n  <div\n    *ngIf=\"open\"\n    class=\"fixed z-50 w-52 rounded-md border border-border bg-popover p-1 text-popover-foreground shadow-md\"\n    [style.left.px]=\"x + 4\"\n    [style.top.px]=\"y + 2\"\n  >\n    <div>\n      <ng-container *ngFor=\"let item of items\">\n        <div *ngIf=\"item.type === 'separator'\" class=\"-mx-1 my-1 h-px bg-muted\"></div>\n\n        <div *ngIf=\"item.type === 'label'\" class=\"px-2 py-1.5 text-sm font-semibold text-foreground\">\n          {{ item.label }}\n        </div>\n\n        <button\n          *ngIf=\"!item.type || item.type === 'item'\"\n          type=\"button\"\n          [disabled]=\"item.disabled\"\n          class=\"relative flex w-full appearance-none cursor-default select-none items-center rounded-sm border-0 bg-transparent py-1.5 pr-2 text-left text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground focus-visible:bg-accent focus-visible:text-accent-foreground disabled:pointer-events-none disabled:opacity-50\"\n          [ngClass]=\"item.inset ? 'pl-8' : 'px-2'\"\n          (click)=\"select(item)\"\n        >\n          <span class=\"mr-2 inline-flex w-4 shrink-0 items-center justify-center text-foreground\">\n            <svg *ngIf=\"item.checked\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n              <path d=\"M5 12.5L9.2 16.7L19 7\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n            </svg>\n            <span *ngIf=\"item.selectedDot\" class=\"h-2 w-2 rounded-full bg-foreground\"></span>\n          </span>\n          <span class=\"min-w-0 flex-1 truncate text-foreground\">{{ item.label }}</span>\n          <span *ngIf=\"item.shortcut\" class=\"text-xs text-muted-foreground\">{{ item.shortcut }}</span>\n          <svg *ngIf=\"item.showChevron\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5 text-muted-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n            <path d=\"M9 6L15 12L9 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n          </svg>\n        </button>\n      </ng-container>\n    </div>\n  </div>\n</div>\n"]}
86
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-menu.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/context-menu/context-menu.component.ts","../../../../../../src/lib/components/context-menu/context-menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output\n} from '@angular/core';\nimport { PdmMenuItem } from '../dropdown-menu/dropdown-menu.component';\n\nexport interface PdmContextMenuItem extends PdmMenuItem {\n  checked?: boolean;\n  selectedDot?: boolean;\n}\n\n@Component({\n  selector: 'pdm-context-menu',\n  templateUrl: './context-menu.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PdmContextMenuComponent implements OnInit, OnDestroy {\n  @Input() items: PdmContextMenuItem[] = [\n    { type: 'item', label: 'Back', value: 'back', inset: true, shortcut: '⌘[' },\n    { type: 'item', label: 'Forward', value: 'forward', inset: true, shortcut: '⌘]', disabled: true },\n    { type: 'item', label: 'Reload', value: 'reload', inset: true, shortcut: '⌘R' },\n    { type: 'item', label: 'More Tools', value: 'more-tools', inset: true, showChevron: true },\n    { type: 'separator' },\n    { type: 'item', label: 'Show Bookmarks Bar', value: 'show-bookmarks', checked: true },\n    { type: 'item', label: 'Show Full URLs', value: 'show-urls', inset: true },\n    { type: 'separator' },\n    { type: 'label', label: 'People' },\n    { type: 'separator' },\n    { type: 'item', label: 'Pedro Duarte', value: 'pedro', selectedDot: true },\n    { type: 'item', label: 'Colm Tuite', value: 'colm', inset: true }\n  ];\n  @Input() className = '';\n  @Input() triggerLabel = 'Right click here';\n  @Input() width = 300;\n  @Input() height = 150;\n  @Output() itemSelect = new EventEmitter<string>();\n\n  open = false;\n  x = 0;\n  y = 0;\n\n  private boundPointerDown!: (event: PointerEvent) => void;\n\n  constructor(private readonly elementRef: ElementRef<HTMLElement>) {}\n\n  ngOnInit(): void {\n    this.boundPointerDown = (event: PointerEvent) => this.onDocumentPointerDown(event);\n    document.addEventListener('pointerdown', this.boundPointerDown, { capture: true });\n  }\n\n  ngOnDestroy(): void {\n    if (this.boundPointerDown) {\n      document.removeEventListener('pointerdown', this.boundPointerDown, { capture: true });\n    }\n  }\n\n  onContextMenu(event: MouseEvent): void {\n    event.preventDefault();\n    this.x = event.clientX;\n    this.y = event.clientY;\n    this.open = true;\n  }\n\n  select(item: PdmContextMenuItem): void {\n    if (item.disabled || item.type === 'separator' || item.type === 'label' || !item.value) return;\n    this.itemSelect.emit(item.value);\n    this.open = false;\n  }\n\n  @HostListener('document:keydown.escape')\n  onEsc(): void {\n    if (this.open) {\n      this.open = false;\n    }\n  }\n\n  private onDocumentPointerDown(event: PointerEvent): void {\n    if (!this.open) return;\n    const target = event.target as Node | null;\n    if (target && !this.elementRef.nativeElement.contains(target)) {\n      this.open = false;\n    }\n  }\n}\n","<div class=\"relative\" [ngClass]=\"className\" (contextmenu)=\"onContextMenu($event)\">\n  <div\n    class=\"flex items-center justify-center rounded-md border border-dashed border-border\"\n    [style.width.px]=\"width\"\n    [style.height.px]=\"height\"\n  >\n    <span class=\"text-sm font-medium text-foreground\">{{ triggerLabel }}</span>\n  </div>\n\n  <div\n    *ngIf=\"open\"\n    class=\"fixed z-[70] min-w-48 max-w-xs rounded-md border border-border bg-popover p-1 text-popover-foreground shadow-md sm:min-w-52\"\n    [style.left.px]=\"x + 4\"\n    [style.top.px]=\"y + 2\"\n  >\n    <div>\n      <ng-container *ngFor=\"let item of items\">\n        <div *ngIf=\"item.type === 'separator'\" class=\"-mx-1 my-1 h-px bg-muted\"></div>\n\n        <div *ngIf=\"item.type === 'label'\" class=\"px-2 py-1.5 text-sm font-semibold text-foreground\">\n          {{ item.label }}\n        </div>\n\n        <button\n          *ngIf=\"!item.type || item.type === 'item'\"\n          type=\"button\"\n          [disabled]=\"item.disabled\"\n          class=\"relative flex w-full appearance-none cursor-default select-none items-center rounded-sm border-0 bg-transparent py-1.5 pr-2 text-left text-sm outline-none transition-colors hover:bg-accent hover:text-accent-foreground focus-visible:bg-accent focus-visible:text-accent-foreground disabled:pointer-events-none disabled:opacity-50\"\n          [ngClass]=\"item.inset ? 'pl-8' : 'px-2'\"\n          (click)=\"select(item)\"\n        >\n          <span class=\"mr-2 inline-flex w-4 shrink-0 items-center justify-center text-foreground\">\n            <svg *ngIf=\"item.checked\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n              <path d=\"M5 12.5L9.2 16.7L19 7\" stroke=\"currentColor\" stroke-width=\"1.8\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n            </svg>\n            <span *ngIf=\"item.selectedDot\" class=\"h-2 w-2 rounded-full bg-foreground\"></span>\n          </span>\n          <span class=\"min-w-0 flex-1 truncate text-foreground\">{{ item.label }}</span>\n          <span *ngIf=\"item.shortcut\" class=\"text-xs text-muted-foreground\">{{ item.shortcut }}</span>\n          <svg *ngIf=\"item.showChevron\" viewBox=\"0 0 24 24\" class=\"h-3.5 w-3.5 text-muted-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n            <path d=\"M9 6L15 12L9 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n          </svg>\n        </button>\n      </ng-container>\n    </div>\n  </div>\n</div>\n"]}
@@ -1,31 +1,148 @@
1
1
  import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
2
2
  import * as i0 from "@angular/core";
3
3
  import * as i1 from "@angular/common";
4
+ import * as i2 from "../table/table.component";
5
+ /**
6
+ * Data-table genérico con paginación, filtrado y selección
7
+ *
8
+ * NUEVO: Ahora es genérico y configurable via columnas
9
+ *
10
+ * @example
11
+ * // Definir columnas
12
+ * columns: PdmDataTableColumn<User>[] = [
13
+ * { key: 'name', label: 'Name', sortable: true },
14
+ * { key: 'email', label: 'Email', sortable: true },
15
+ * { key: 'role', label: 'Role', hideOnMobile: true },
16
+ * { key: 'createdAt', label: 'Created', render: (val) => formatDate(val) }
17
+ * ];
18
+ *
19
+ * // En el template
20
+ * <pdm-data-table
21
+ * [columns]="columns"
22
+ * [rows]="users"
23
+ * [selectable]="true"
24
+ * (selectionChange)="onSelect($event)">
25
+ * </pdm-data-table>
26
+ */
4
27
  export class PdmDataTableComponent {
5
28
  constructor() {
6
29
  this.className = '';
30
+ /**
31
+ * Columnas a mostrar
32
+ * Si no se provee, intenta inferir del primer row (legacy mode)
33
+ */
34
+ this.columns = [];
35
+ /**
36
+ * Estrategia responsive de la tabla
37
+ */
38
+ this.responsiveStrategy = 'scroll';
39
+ /**
40
+ * Si es true, muestra checkbox de selección en cada fila
41
+ */
42
+ this.selectable = false;
43
+ /**
44
+ * Si es true, muestra botón de acciones (tres puntos) en cada fila
45
+ */
46
+ this.showActions = false;
47
+ /**
48
+ * Si es true, muestra filtro de búsqueda
49
+ */
50
+ this.showFilter = true;
51
+ /**
52
+ * Si es true, muestra controles de paginación
53
+ */
54
+ this.showPagination = true;
55
+ /**
56
+ * Si es true, muestra selector de columnas
57
+ */
58
+ this.showColumnSelector = false;
59
+ // Labels i18n
7
60
  this.filterPlaceholder = 'Filter...';
8
61
  this.columnsLabel = 'Columns';
9
- this.statusLabel = 'Status';
10
- this.emailLabel = 'Email';
11
- this.amountLabel = 'Amount';
12
62
  this.previousLabel = 'Previous';
13
63
  this.nextLabel = 'Next';
14
64
  this.emptyLabel = 'No results.';
65
+ this.rowsSelectedLabel = 'row(s) selected';
66
+ // DEPRECATED: Labels hardcodeados para backward compatibility
67
+ /**
68
+ * @deprecated Use columns configuration instead
69
+ */
70
+ this.statusLabel = 'Status';
71
+ /**
72
+ * @deprecated Use columns configuration instead
73
+ */
74
+ this.emailLabel = 'Email';
75
+ /**
76
+ * @deprecated Use columns configuration instead
77
+ */
78
+ this.amountLabel = 'Amount';
79
+ /**
80
+ * Datos a mostrar
81
+ */
15
82
  this.rows = [];
83
+ /**
84
+ * Página actual (1-indexed)
85
+ */
16
86
  this.page = 1;
17
- this.pageSize = 5;
87
+ /**
88
+ * Cantidad de filas por página
89
+ */
90
+ this.pageSize = 10;
91
+ /**
92
+ * Query de filtrado
93
+ */
18
94
  this.query = '';
19
95
  this.queryChange = new EventEmitter();
20
96
  this.rowAction = new EventEmitter();
21
97
  this.pageChange = new EventEmitter();
22
98
  this.selectionChange = new EventEmitter();
99
+ this.columnSort = new EventEmitter();
100
+ // Estado interno
101
+ this.selectedRows = new Set();
102
+ this.sortDirection = 'asc';
103
+ }
104
+ /**
105
+ * Backward compatibility: si no hay columnas definidas, inferir del primer row
106
+ */
107
+ get effectiveColumns() {
108
+ if (this.columns.length > 0) {
109
+ return this.columns;
110
+ }
111
+ // Legacy mode: inferir columnas del primer row (solo para PdmDataTableRow)
112
+ if (this.rows.length > 0) {
113
+ const firstRow = this.rows[0];
114
+ return Object.keys(firstRow)
115
+ .filter(key => key !== 'selected')
116
+ .map(key => ({
117
+ key: key,
118
+ label: this.getLegacyLabel(key),
119
+ align: key === 'amount' ? 'right' : 'left'
120
+ }));
121
+ }
122
+ return [];
123
+ }
124
+ /**
125
+ * LEGACY: mapeo de keys a labels hardcodeados
126
+ */
127
+ getLegacyLabel(key) {
128
+ const map = {
129
+ status: this.statusLabel,
130
+ email: this.emailLabel,
131
+ amount: this.amountLabel
132
+ };
133
+ return map[key] || key.charAt(0).toUpperCase() + key.slice(1);
23
134
  }
24
135
  get filteredRows() {
25
136
  const q = this.query.trim().toLowerCase();
26
137
  if (!q)
27
138
  return this.rows;
28
- return this.rows.filter((r) => r.email.toLowerCase().includes(q));
139
+ if (this.filterFn) {
140
+ return this.rows.filter(row => this.filterFn(row, q));
141
+ }
142
+ // Filtrado default: buscar en todos los campos string
143
+ return this.rows.filter(row => {
144
+ return Object.values(row).some(val => typeof val === 'string' && val.toLowerCase().includes(q));
145
+ });
29
146
  }
30
147
  get pagedRows() {
31
148
  const start = (this.page - 1) * this.pageSize;
@@ -35,14 +152,33 @@ export class PdmDataTableComponent {
35
152
  return Math.max(1, Math.ceil(this.filteredRows.length / this.pageSize));
36
153
  }
37
154
  get selectedCount() {
38
- return this.rows.filter((row) => row.selected).length;
155
+ return this.selectedRows.size;
39
156
  }
40
157
  onQueryInput(event) {
41
158
  const value = event.target.value;
42
159
  this.queryChange.emit(value);
43
160
  }
44
161
  onToggleRow(row, event) {
45
- this.selectionChange.emit({ id: row.id, selected: event.target.checked });
162
+ const checked = event.target.checked;
163
+ if (checked) {
164
+ this.selectedRows.add(row);
165
+ }
166
+ else {
167
+ this.selectedRows.delete(row);
168
+ }
169
+ this.selectionChange.emit({ row, selected: checked });
170
+ }
171
+ onToggleAll(event) {
172
+ const checked = event.target.checked;
173
+ if (checked) {
174
+ this.pagedRows.forEach(row => this.selectedRows.add(row));
175
+ }
176
+ else {
177
+ this.pagedRows.forEach(row => this.selectedRows.delete(row));
178
+ }
179
+ }
180
+ isSelected(row) {
181
+ return this.selectedRows.has(row);
46
182
  }
47
183
  previous() {
48
184
  if (this.page <= 1)
@@ -55,25 +191,75 @@ export class PdmDataTableComponent {
55
191
  this.pageChange.emit(this.page + 1);
56
192
  }
57
193
  onAction(row) {
58
- this.rowAction.emit(row.id);
194
+ this.rowAction.emit(row);
195
+ }
196
+ onSort(column) {
197
+ if (!column.sortable)
198
+ return;
199
+ if (this.sortColumn === column) {
200
+ this.sortDirection = this.sortDirection === 'asc' ? 'desc' : 'asc';
201
+ }
202
+ else {
203
+ this.sortColumn = column;
204
+ this.sortDirection = 'asc';
205
+ }
206
+ this.columnSort.emit({ column, direction: this.sortDirection });
207
+ }
208
+ getCellValue(row, column) {
209
+ const value = row[column.key];
210
+ if (column.render) {
211
+ return column.render(value, row);
212
+ }
213
+ return value != null ? String(value) : '';
214
+ }
215
+ getCellClass(column) {
216
+ const classes = ['px-2', 'py-2'];
217
+ if (column.align === 'center')
218
+ classes.push('text-center');
219
+ if (column.align === 'right')
220
+ classes.push('text-right');
221
+ if (column.hideOnMobile)
222
+ classes.push('hidden', 'md:table-cell');
223
+ if (column.cellClass)
224
+ classes.push(column.cellClass);
225
+ return classes.join(' ');
226
+ }
227
+ getHeaderClass(column) {
228
+ const classes = ['px-2', 'py-2', 'text-left', 'font-medium'];
229
+ if (column.hideOnMobile)
230
+ classes.push('hidden', 'md:table-cell');
231
+ if (column.headerClass)
232
+ classes.push(column.headerClass);
233
+ return classes.join(' ');
234
+ }
235
+ getColumnStyle(column) {
236
+ return column.width ? { width: column.width } : {};
59
237
  }
60
238
  }
61
239
  PdmDataTableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmDataTableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
62
- PdmDataTableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PdmDataTableComponent, selector: "pdm-data-table", inputs: { className: "className", filterPlaceholder: "filterPlaceholder", columnsLabel: "columnsLabel", statusLabel: "statusLabel", emailLabel: "emailLabel", amountLabel: "amountLabel", previousLabel: "previousLabel", nextLabel: "nextLabel", emptyLabel: "emptyLabel", rows: "rows", page: "page", pageSize: "pageSize", query: "query" }, outputs: { queryChange: "queryChange", rowAction: "rowAction", pageChange: "pageChange", selectionChange: "selectionChange" }, ngImport: i0, template: "<section [ngClass]=\"['flex w-full max-w-3xl flex-col items-end', className]\">\n <div class=\"flex w-full items-center justify-between py-4\">\n <input\n type=\"text\"\n [placeholder]=\"filterPlaceholder\"\n [value]=\"query\"\n (input)=\"onQueryInput($event)\"\n class=\"h-9 flex-1 rounded-md border border-input bg-transparent px-3 py-1 text-sm text-foreground shadow-sm placeholder:text-muted-foreground outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\"\n />\n\n <button type=\"button\" class=\"inline-flex h-9 appearance-none items-center gap-2 rounded-md border border-input bg-background px-3 py-2 text-sm font-medium text-foreground shadow-sm\">\n <span>{{ columnsLabel }}</span>\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4 text-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M7 10L12 15L17 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </div>\n\n <div class=\"w-full overflow-hidden rounded-md border border-border bg-background\">\n <table class=\"w-full border-collapse text-sm text-foreground\">\n <thead>\n <tr class=\"border-b border-border\">\n <th class=\"w-10 px-2 text-left font-medium\"><input type=\"checkbox\" class=\"h-4 w-4 rounded-sm border border-input\" /></th>\n <th class=\"w-32 px-2 py-2 text-left font-medium\">{{ statusLabel }}</th>\n <th class=\"px-2 py-2 text-left font-medium\">\n <button type=\"button\" class=\"inline-flex appearance-none items-center gap-1 rounded-sm border-0 bg-transparent px-3 py-2 text-sm\">\n <span>{{ emailLabel }}</span>\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M8 6L4 10L8 14M16 18L20 14L16 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </th>\n <th class=\"w-24 px-2 py-2 text-right font-medium\">{{ amountLabel }}</th>\n <th class=\"px-2 py-2\"></th>\n </tr>\n </thead>\n\n <tbody>\n <tr *ngFor=\"let row of pagedRows\" class=\"border-b border-border last:border-b-0\">\n <td class=\"px-2 py-2\"><input type=\"checkbox\" [checked]=\"row.selected\" (change)=\"onToggleRow(row, $event)\" class=\"h-4 w-4 rounded-sm border border-input\" /></td>\n <td class=\"px-2 py-2\">{{ row.status }}</td>\n <td class=\"px-2 py-2\">{{ row.email }}</td>\n <td class=\"px-2 py-2 text-right\">{{ row.amount }}</td>\n <td class=\"px-2 py-2\">\n <button type=\"button\" class=\"inline-flex h-8 w-8 appearance-none items-center justify-center border-0 bg-transparent p-0\" (click)=\"onAction(row)\">\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"6\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n <circle cx=\"12\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n <circle cx=\"18\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n </svg>\n </button>\n </td>\n </tr>\n <tr *ngIf=\"pagedRows.length === 0\">\n <td colspan=\"5\" class=\"px-3 py-6 text-center text-sm text-muted-foreground\">{{ emptyLabel }}</td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div class=\"flex w-full items-center gap-2 py-4\">\n <p class=\"m-0 flex-1 pr-2 text-sm text-muted-foreground\">{{ selectedCount }} of {{ rows.length }} row(s) selected.</p>\n <button type=\"button\" class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50\" [disabled]=\"page <= 1\" (click)=\"previous()\">{{ previousLabel }}</button>\n <button type=\"button\" class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50\" [disabled]=\"page >= totalPages\" (click)=\"next()\">{{ nextLabel }}</button>\n </div>\n</section>\n", dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
240
+ PdmDataTableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PdmDataTableComponent, selector: "pdm-data-table", inputs: { className: "className", columns: "columns", responsiveStrategy: "responsiveStrategy", selectable: "selectable", showActions: "showActions", showFilter: "showFilter", showPagination: "showPagination", showColumnSelector: "showColumnSelector", filterPlaceholder: "filterPlaceholder", columnsLabel: "columnsLabel", previousLabel: "previousLabel", nextLabel: "nextLabel", emptyLabel: "emptyLabel", rowsSelectedLabel: "rowsSelectedLabel", statusLabel: "statusLabel", emailLabel: "emailLabel", amountLabel: "amountLabel", rows: "rows", page: "page", pageSize: "pageSize", query: "query", filterFn: "filterFn" }, outputs: { queryChange: "queryChange", rowAction: "rowAction", pageChange: "pageChange", selectionChange: "selectionChange", columnSort: "columnSort" }, ngImport: i0, template: "<section [ngClass]=\"['flex w-full flex-col', className]\">\n <!-- Toolbar: Filtro + Selector de columnas -->\n <div *ngIf=\"showFilter || showColumnSelector\" class=\"flex w-full items-center justify-between gap-2 py-4\">\n <input\n *ngIf=\"showFilter\"\n type=\"text\"\n [placeholder]=\"filterPlaceholder\"\n [value]=\"query\"\n (input)=\"onQueryInput($event)\"\n class=\"h-9 flex-1 rounded-md border border-input bg-transparent px-3 py-1 text-sm text-foreground shadow-sm placeholder:text-muted-foreground outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\"\n />\n\n <button \n *ngIf=\"showColumnSelector\"\n type=\"button\" \n class=\"inline-flex h-9 appearance-none items-center gap-2 rounded-md border border-input bg-background px-3 py-2 text-sm font-medium text-foreground shadow-sm whitespace-nowrap\">\n <span>{{ columnsLabel }}</span>\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4 text-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M7 10L12 15L17 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </div>\n\n <!-- Tabla con responsive -->\n <pdm-table \n variant=\"data\"\n [responsiveStrategy]=\"responsiveStrategy\"\n [fullBleed]=\"false\">\n <thead>\n <tr>\n <!-- Columna de selecci\u00F3n -->\n <th *ngIf=\"selectable\" class=\"w-10 px-2 py-2 text-left font-medium\">\n <input \n type=\"checkbox\" \n (change)=\"onToggleAll($event)\"\n class=\"h-4 w-4 rounded-sm border border-input\" \n />\n </th>\n\n <!-- Columnas din\u00E1micas -->\n <th \n *ngFor=\"let column of effectiveColumns\"\n [ngClass]=\"getHeaderClass(column)\"\n [ngStyle]=\"getColumnStyle(column)\">\n \n <!-- Header sortable -->\n <button \n *ngIf=\"column.sortable\"\n type=\"button\" \n (click)=\"onSort(column)\"\n class=\"inline-flex appearance-none items-center gap-1 rounded-sm border-0 bg-transparent px-3 py-2 text-sm hover:underline\">\n <span>{{ column.label }}</span>\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M8 6L4 10L8 14M16 18L20 14L16 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n\n <!-- Header no sortable -->\n <span *ngIf=\"!column.sortable\">{{ column.label }}</span>\n </th>\n\n <!-- Columna de acciones -->\n <th *ngIf=\"showActions\" class=\"w-10 px-2 py-2\"></th>\n </tr>\n </thead>\n\n <tbody>\n <!-- Filas con datos -->\n <tr *ngFor=\"let row of pagedRows\">\n <!-- Celda de selecci\u00F3n -->\n <td *ngIf=\"selectable\" class=\"px-2 py-2\">\n <input \n type=\"checkbox\" \n [checked]=\"isSelected(row)\" \n (change)=\"onToggleRow(row, $event)\" \n class=\"h-4 w-4 rounded-sm border border-input\" \n />\n </td>\n\n <!-- Celdas din\u00E1micas -->\n <td \n *ngFor=\"let column of effectiveColumns\"\n [ngClass]=\"getCellClass(column)\">\n \n <!-- Template personalizado si existe -->\n <ng-container *ngIf=\"column.cellTemplate; else defaultCell\">\n <ng-container \n *ngTemplateOutlet=\"column.cellTemplate; context: { $implicit: row, value: row[column.key] }\">\n </ng-container>\n </ng-container>\n\n <!-- Renderizado default -->\n <ng-template #defaultCell>\n {{ getCellValue(row, column) }}\n </ng-template>\n </td>\n\n <!-- Celda de acciones -->\n <td *ngIf=\"showActions\" class=\"px-2 py-2\">\n <button \n type=\"button\" \n class=\"inline-flex h-8 w-8 appearance-none items-center justify-center border-0 bg-transparent p-0 hover:text-foreground\" \n (click)=\"onAction(row)\">\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"6\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n <circle cx=\"12\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n <circle cx=\"18\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n </svg>\n </button>\n </td>\n </tr>\n\n <!-- Fila vac\u00EDa -->\n <tr *ngIf=\"pagedRows.length === 0\">\n <td \n [attr.colspan]=\"effectiveColumns.length + (selectable ? 1 : 0) + (showActions ? 1 : 0)\" \n class=\"px-3 py-6 text-center text-sm text-muted-foreground\">\n {{ emptyLabel }}\n </td>\n </tr>\n </tbody>\n </pdm-table>\n\n <!-- Footer: Info + Paginaci\u00F3n -->\n <div *ngIf=\"showPagination || selectable\" class=\"flex w-full items-center gap-2 py-4 flex-wrap sm:flex-nowrap\">\n <p *ngIf=\"selectable\" class=\"m-0 flex-1 pr-2 text-sm text-muted-foreground whitespace-nowrap\">\n {{ selectedCount }} of {{ rows.length }} {{ rowsSelectedLabel }}\n </p>\n\n <div *ngIf=\"showPagination\" class=\"flex items-center gap-2 ml-auto\">\n <span class=\"text-sm text-muted-foreground whitespace-nowrap\">\n Page {{ page }} of {{ totalPages }}\n </span>\n <button \n type=\"button\" \n class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:cursor-not-allowed\" \n [disabled]=\"page <= 1\" \n (click)=\"previous()\">\n {{ previousLabel }}\n </button>\n <button \n type=\"button\" \n class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:cursor-not-allowed\" \n [disabled]=\"page >= totalPages\" \n (click)=\"next()\">\n {{ nextLabel }}\n </button>\n </div>\n </div>\n</section>\n", dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i2.PdmTableComponent, selector: "pdm-table", inputs: ["variant", "responsiveStrategy", "className", "fullBleed"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
63
241
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmDataTableComponent, decorators: [{
64
242
  type: Component,
65
- args: [{ selector: 'pdm-data-table', changeDetection: ChangeDetectionStrategy.OnPush, template: "<section [ngClass]=\"['flex w-full max-w-3xl flex-col items-end', className]\">\n <div class=\"flex w-full items-center justify-between py-4\">\n <input\n type=\"text\"\n [placeholder]=\"filterPlaceholder\"\n [value]=\"query\"\n (input)=\"onQueryInput($event)\"\n class=\"h-9 flex-1 rounded-md border border-input bg-transparent px-3 py-1 text-sm text-foreground shadow-sm placeholder:text-muted-foreground outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\"\n />\n\n <button type=\"button\" class=\"inline-flex h-9 appearance-none items-center gap-2 rounded-md border border-input bg-background px-3 py-2 text-sm font-medium text-foreground shadow-sm\">\n <span>{{ columnsLabel }}</span>\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4 text-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M7 10L12 15L17 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </div>\n\n <div class=\"w-full overflow-hidden rounded-md border border-border bg-background\">\n <table class=\"w-full border-collapse text-sm text-foreground\">\n <thead>\n <tr class=\"border-b border-border\">\n <th class=\"w-10 px-2 text-left font-medium\"><input type=\"checkbox\" class=\"h-4 w-4 rounded-sm border border-input\" /></th>\n <th class=\"w-32 px-2 py-2 text-left font-medium\">{{ statusLabel }}</th>\n <th class=\"px-2 py-2 text-left font-medium\">\n <button type=\"button\" class=\"inline-flex appearance-none items-center gap-1 rounded-sm border-0 bg-transparent px-3 py-2 text-sm\">\n <span>{{ emailLabel }}</span>\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M8 6L4 10L8 14M16 18L20 14L16 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </th>\n <th class=\"w-24 px-2 py-2 text-right font-medium\">{{ amountLabel }}</th>\n <th class=\"px-2 py-2\"></th>\n </tr>\n </thead>\n\n <tbody>\n <tr *ngFor=\"let row of pagedRows\" class=\"border-b border-border last:border-b-0\">\n <td class=\"px-2 py-2\"><input type=\"checkbox\" [checked]=\"row.selected\" (change)=\"onToggleRow(row, $event)\" class=\"h-4 w-4 rounded-sm border border-input\" /></td>\n <td class=\"px-2 py-2\">{{ row.status }}</td>\n <td class=\"px-2 py-2\">{{ row.email }}</td>\n <td class=\"px-2 py-2 text-right\">{{ row.amount }}</td>\n <td class=\"px-2 py-2\">\n <button type=\"button\" class=\"inline-flex h-8 w-8 appearance-none items-center justify-center border-0 bg-transparent p-0\" (click)=\"onAction(row)\">\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"6\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n <circle cx=\"12\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n <circle cx=\"18\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n </svg>\n </button>\n </td>\n </tr>\n <tr *ngIf=\"pagedRows.length === 0\">\n <td colspan=\"5\" class=\"px-3 py-6 text-center text-sm text-muted-foreground\">{{ emptyLabel }}</td>\n </tr>\n </tbody>\n </table>\n </div>\n\n <div class=\"flex w-full items-center gap-2 py-4\">\n <p class=\"m-0 flex-1 pr-2 text-sm text-muted-foreground\">{{ selectedCount }} of {{ rows.length }} row(s) selected.</p>\n <button type=\"button\" class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50\" [disabled]=\"page <= 1\" (click)=\"previous()\">{{ previousLabel }}</button>\n <button type=\"button\" class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50\" [disabled]=\"page >= totalPages\" (click)=\"next()\">{{ nextLabel }}</button>\n </div>\n</section>\n" }]
243
+ args: [{ selector: 'pdm-data-table', changeDetection: ChangeDetectionStrategy.OnPush, template: "<section [ngClass]=\"['flex w-full flex-col', className]\">\n <!-- Toolbar: Filtro + Selector de columnas -->\n <div *ngIf=\"showFilter || showColumnSelector\" class=\"flex w-full items-center justify-between gap-2 py-4\">\n <input\n *ngIf=\"showFilter\"\n type=\"text\"\n [placeholder]=\"filterPlaceholder\"\n [value]=\"query\"\n (input)=\"onQueryInput($event)\"\n class=\"h-9 flex-1 rounded-md border border-input bg-transparent px-3 py-1 text-sm text-foreground shadow-sm placeholder:text-muted-foreground outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\"\n />\n\n <button \n *ngIf=\"showColumnSelector\"\n type=\"button\" \n class=\"inline-flex h-9 appearance-none items-center gap-2 rounded-md border border-input bg-background px-3 py-2 text-sm font-medium text-foreground shadow-sm whitespace-nowrap\">\n <span>{{ columnsLabel }}</span>\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4 text-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M7 10L12 15L17 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n </div>\n\n <!-- Tabla con responsive -->\n <pdm-table \n variant=\"data\"\n [responsiveStrategy]=\"responsiveStrategy\"\n [fullBleed]=\"false\">\n <thead>\n <tr>\n <!-- Columna de selecci\u00F3n -->\n <th *ngIf=\"selectable\" class=\"w-10 px-2 py-2 text-left font-medium\">\n <input \n type=\"checkbox\" \n (change)=\"onToggleAll($event)\"\n class=\"h-4 w-4 rounded-sm border border-input\" \n />\n </th>\n\n <!-- Columnas din\u00E1micas -->\n <th \n *ngFor=\"let column of effectiveColumns\"\n [ngClass]=\"getHeaderClass(column)\"\n [ngStyle]=\"getColumnStyle(column)\">\n \n <!-- Header sortable -->\n <button \n *ngIf=\"column.sortable\"\n type=\"button\" \n (click)=\"onSort(column)\"\n class=\"inline-flex appearance-none items-center gap-1 rounded-sm border-0 bg-transparent px-3 py-2 text-sm hover:underline\">\n <span>{{ column.label }}</span>\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M8 6L4 10L8 14M16 18L20 14L16 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n </svg>\n </button>\n\n <!-- Header no sortable -->\n <span *ngIf=\"!column.sortable\">{{ column.label }}</span>\n </th>\n\n <!-- Columna de acciones -->\n <th *ngIf=\"showActions\" class=\"w-10 px-2 py-2\"></th>\n </tr>\n </thead>\n\n <tbody>\n <!-- Filas con datos -->\n <tr *ngFor=\"let row of pagedRows\">\n <!-- Celda de selecci\u00F3n -->\n <td *ngIf=\"selectable\" class=\"px-2 py-2\">\n <input \n type=\"checkbox\" \n [checked]=\"isSelected(row)\" \n (change)=\"onToggleRow(row, $event)\" \n class=\"h-4 w-4 rounded-sm border border-input\" \n />\n </td>\n\n <!-- Celdas din\u00E1micas -->\n <td \n *ngFor=\"let column of effectiveColumns\"\n [ngClass]=\"getCellClass(column)\">\n \n <!-- Template personalizado si existe -->\n <ng-container *ngIf=\"column.cellTemplate; else defaultCell\">\n <ng-container \n *ngTemplateOutlet=\"column.cellTemplate; context: { $implicit: row, value: row[column.key] }\">\n </ng-container>\n </ng-container>\n\n <!-- Renderizado default -->\n <ng-template #defaultCell>\n {{ getCellValue(row, column) }}\n </ng-template>\n </td>\n\n <!-- Celda de acciones -->\n <td *ngIf=\"showActions\" class=\"px-2 py-2\">\n <button \n type=\"button\" \n class=\"inline-flex h-8 w-8 appearance-none items-center justify-center border-0 bg-transparent p-0 hover:text-foreground\" \n (click)=\"onAction(row)\">\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <circle cx=\"6\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n <circle cx=\"12\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n <circle cx=\"18\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n </svg>\n </button>\n </td>\n </tr>\n\n <!-- Fila vac\u00EDa -->\n <tr *ngIf=\"pagedRows.length === 0\">\n <td \n [attr.colspan]=\"effectiveColumns.length + (selectable ? 1 : 0) + (showActions ? 1 : 0)\" \n class=\"px-3 py-6 text-center text-sm text-muted-foreground\">\n {{ emptyLabel }}\n </td>\n </tr>\n </tbody>\n </pdm-table>\n\n <!-- Footer: Info + Paginaci\u00F3n -->\n <div *ngIf=\"showPagination || selectable\" class=\"flex w-full items-center gap-2 py-4 flex-wrap sm:flex-nowrap\">\n <p *ngIf=\"selectable\" class=\"m-0 flex-1 pr-2 text-sm text-muted-foreground whitespace-nowrap\">\n {{ selectedCount }} of {{ rows.length }} {{ rowsSelectedLabel }}\n </p>\n\n <div *ngIf=\"showPagination\" class=\"flex items-center gap-2 ml-auto\">\n <span class=\"text-sm text-muted-foreground whitespace-nowrap\">\n Page {{ page }} of {{ totalPages }}\n </span>\n <button \n type=\"button\" \n class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:cursor-not-allowed\" \n [disabled]=\"page <= 1\" \n (click)=\"previous()\">\n {{ previousLabel }}\n </button>\n <button \n type=\"button\" \n class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:cursor-not-allowed\" \n [disabled]=\"page >= totalPages\" \n (click)=\"next()\">\n {{ nextLabel }}\n </button>\n </div>\n </div>\n</section>\n" }]
66
244
  }], propDecorators: { className: [{
67
245
  type: Input
68
- }], filterPlaceholder: [{
246
+ }], columns: [{
69
247
  type: Input
70
- }], columnsLabel: [{
248
+ }], responsiveStrategy: [{
71
249
  type: Input
72
- }], statusLabel: [{
250
+ }], selectable: [{
73
251
  type: Input
74
- }], emailLabel: [{
252
+ }], showActions: [{
75
253
  type: Input
76
- }], amountLabel: [{
254
+ }], showFilter: [{
255
+ type: Input
256
+ }], showPagination: [{
257
+ type: Input
258
+ }], showColumnSelector: [{
259
+ type: Input
260
+ }], filterPlaceholder: [{
261
+ type: Input
262
+ }], columnsLabel: [{
77
263
  type: Input
78
264
  }], previousLabel: [{
79
265
  type: Input
@@ -81,6 +267,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
81
267
  type: Input
82
268
  }], emptyLabel: [{
83
269
  type: Input
270
+ }], rowsSelectedLabel: [{
271
+ type: Input
272
+ }], statusLabel: [{
273
+ type: Input
274
+ }], emailLabel: [{
275
+ type: Input
276
+ }], amountLabel: [{
277
+ type: Input
84
278
  }], rows: [{
85
279
  type: Input
86
280
  }], page: [{
@@ -89,6 +283,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
89
283
  type: Input
90
284
  }], query: [{
91
285
  type: Input
286
+ }], filterFn: [{
287
+ type: Input
92
288
  }], queryChange: [{
93
289
  type: Output
94
290
  }], rowAction: [{
@@ -97,5 +293,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
97
293
  type: Output
98
294
  }], selectionChange: [{
99
295
  type: Output
296
+ }], columnSort: [{
297
+ type: Output
100
298
  }] } });
101
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-table.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/data-table/data-table.component.ts","../../../../../../src/lib/components/data-table/data-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';\n\nexport interface PdmDataTableRow {\n  id: string;\n  status: string;\n  email: string;\n  amount: string;\n  selected?: boolean;\n}\n\n@Component({\n  selector: 'pdm-data-table',\n  templateUrl: './data-table.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PdmDataTableComponent {\n  @Input() className = '';\n  @Input() filterPlaceholder = 'Filter...';\n  @Input() columnsLabel = 'Columns';\n  @Input() statusLabel = 'Status';\n  @Input() emailLabel = 'Email';\n  @Input() amountLabel = 'Amount';\n  @Input() previousLabel = 'Previous';\n  @Input() nextLabel = 'Next';\n  @Input() emptyLabel = 'No results.';\n  @Input() rows: PdmDataTableRow[] = [];\n\n  @Input() page = 1;\n  @Input() pageSize = 5;\n  @Input() query = '';\n\n  @Output() queryChange = new EventEmitter<string>();\n  @Output() rowAction = new EventEmitter<string>();\n  @Output() pageChange = new EventEmitter<number>();\n  @Output() selectionChange = new EventEmitter<{ id: string; selected: boolean }>();\n\n  get filteredRows(): PdmDataTableRow[] {\n    const q = this.query.trim().toLowerCase();\n    if (!q) return this.rows;\n    return this.rows.filter((r) => r.email.toLowerCase().includes(q));\n  }\n\n  get pagedRows(): PdmDataTableRow[] {\n    const start = (this.page - 1) * this.pageSize;\n    return this.filteredRows.slice(start, start + this.pageSize);\n  }\n\n  get totalPages(): number {\n    return Math.max(1, Math.ceil(this.filteredRows.length / this.pageSize));\n  }\n\n  get selectedCount(): number {\n    return this.rows.filter((row) => row.selected).length;\n  }\n\n  onQueryInput(event: Event): void {\n    const value = (event.target as HTMLInputElement).value;\n    this.queryChange.emit(value);\n  }\n\n  onToggleRow(row: PdmDataTableRow, event: Event): void {\n    this.selectionChange.emit({ id: row.id, selected: (event.target as HTMLInputElement).checked });\n  }\n\n  previous(): void {\n    if (this.page <= 1) return;\n    this.pageChange.emit(this.page - 1);\n  }\n\n  next(): void {\n    if (this.page >= this.totalPages) return;\n    this.pageChange.emit(this.page + 1);\n  }\n\n  onAction(row: PdmDataTableRow): void {\n    this.rowAction.emit(row.id);\n  }\n}\n","<section [ngClass]=\"['flex w-full max-w-3xl flex-col items-end', className]\">\n  <div class=\"flex w-full items-center justify-between py-4\">\n    <input\n      type=\"text\"\n      [placeholder]=\"filterPlaceholder\"\n      [value]=\"query\"\n      (input)=\"onQueryInput($event)\"\n      class=\"h-9 flex-1 rounded-md border border-input bg-transparent px-3 py-1 text-sm text-foreground shadow-sm placeholder:text-muted-foreground outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\"\n    />\n\n    <button type=\"button\" class=\"inline-flex h-9 appearance-none items-center gap-2 rounded-md border border-input bg-background px-3 py-2 text-sm font-medium text-foreground shadow-sm\">\n      <span>{{ columnsLabel }}</span>\n      <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4 text-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M7 10L12 15L17 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n      </svg>\n    </button>\n  </div>\n\n  <div class=\"w-full overflow-hidden rounded-md border border-border bg-background\">\n    <table class=\"w-full border-collapse text-sm text-foreground\">\n      <thead>\n        <tr class=\"border-b border-border\">\n          <th class=\"w-10 px-2 text-left font-medium\"><input type=\"checkbox\" class=\"h-4 w-4 rounded-sm border border-input\" /></th>\n          <th class=\"w-32 px-2 py-2 text-left font-medium\">{{ statusLabel }}</th>\n          <th class=\"px-2 py-2 text-left font-medium\">\n            <button type=\"button\" class=\"inline-flex appearance-none items-center gap-1 rounded-sm border-0 bg-transparent px-3 py-2 text-sm\">\n              <span>{{ emailLabel }}</span>\n              <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n                <path d=\"M8 6L4 10L8 14M16 18L20 14L16 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n              </svg>\n            </button>\n          </th>\n          <th class=\"w-24 px-2 py-2 text-right font-medium\">{{ amountLabel }}</th>\n          <th class=\"px-2 py-2\"></th>\n        </tr>\n      </thead>\n\n      <tbody>\n        <tr *ngFor=\"let row of pagedRows\" class=\"border-b border-border last:border-b-0\">\n          <td class=\"px-2 py-2\"><input type=\"checkbox\" [checked]=\"row.selected\" (change)=\"onToggleRow(row, $event)\" class=\"h-4 w-4 rounded-sm border border-input\" /></td>\n          <td class=\"px-2 py-2\">{{ row.status }}</td>\n          <td class=\"px-2 py-2\">{{ row.email }}</td>\n          <td class=\"px-2 py-2 text-right\">{{ row.amount }}</td>\n          <td class=\"px-2 py-2\">\n            <button type=\"button\" class=\"inline-flex h-8 w-8 appearance-none items-center justify-center border-0 bg-transparent p-0\" (click)=\"onAction(row)\">\n              <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n                <circle cx=\"6\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n                <circle cx=\"12\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n                <circle cx=\"18\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n              </svg>\n            </button>\n          </td>\n        </tr>\n        <tr *ngIf=\"pagedRows.length === 0\">\n          <td colspan=\"5\" class=\"px-3 py-6 text-center text-sm text-muted-foreground\">{{ emptyLabel }}</td>\n        </tr>\n      </tbody>\n    </table>\n  </div>\n\n  <div class=\"flex w-full items-center gap-2 py-4\">\n    <p class=\"m-0 flex-1 pr-2 text-sm text-muted-foreground\">{{ selectedCount }} of {{ rows.length }} row(s) selected.</p>\n    <button type=\"button\" class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50\" [disabled]=\"page <= 1\" (click)=\"previous()\">{{ previousLabel }}</button>\n    <button type=\"button\" class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50\" [disabled]=\"page >= totalPages\" (click)=\"next()\">{{ nextLabel }}</button>\n  </div>\n</section>\n"]}
299
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-table.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/data-table/data-table.component.ts","../../../../../../src/lib/components/data-table/data-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, TemplateRef } from '@angular/core';\nimport { TableResponsiveStrategy } from '../../utils/responsive';\n\n/**\n * DEPRECATED: Esta interfaz es para backward compatibility\n * Usar PdmDataTableColumn<T> con tipo genérico en su lugar\n */\nexport interface PdmDataTableRow {\n  id: string;\n  status: string;\n  email: string;\n  amount: string;\n  selected?: boolean;\n}\n\n/**\n * Definición de columna para data-table genérico\n */\nexport interface PdmDataTableColumn<T = any> {\n  /**\n   * Key del campo en el objeto de datos\n   * Usado para acceder al valor: row[key]\n   */\n  key: keyof T;\n  \n  /**\n   * Label a mostrar en el header\n   */\n  label: string;\n  \n  /**\n   * Ancho de la columna (CSS width)\n   * Ej: '100px', '20%', 'auto'\n   */\n  width?: string;\n  \n  /**\n   * Si la columna es sortable\n   */\n  sortable?: boolean;\n  \n  /**\n   * Alineación del contenido\n   */\n  align?: 'left' | 'center' | 'right';\n  \n  /**\n   * Función custom para renderizar el valor\n   * Si no se provee, se usa toString() del valor\n   */\n  render?: (value: any, row: T) => string;\n  \n  /**\n   * Template personalizado para la celda\n   * Tiene prioridad sobre render()\n   */\n  cellTemplate?: TemplateRef<{ $implicit: T; value: any }>;\n  \n  /**\n   * Si es true, la columna se oculta en mobile\n   * Solo se muestra en breakpoint md+ (768px)\n   */\n  hideOnMobile?: boolean;\n  \n  /**\n   * CSS classes adicionales para las celdas de esta columna\n   */\n  cellClass?: string;\n  \n  /**\n   * CSS classes adicionales para el header de esta columna\n   */\n  headerClass?: string;\n}\n\n/**\n * Data-table genérico con paginación, filtrado y selección\n * \n * NUEVO: Ahora es genérico y configurable via columnas\n * \n * @example\n * // Definir columnas\n * columns: PdmDataTableColumn<User>[] = [\n *   { key: 'name', label: 'Name', sortable: true },\n *   { key: 'email', label: 'Email', sortable: true },\n *   { key: 'role', label: 'Role', hideOnMobile: true },\n *   { key: 'createdAt', label: 'Created', render: (val) => formatDate(val) }\n * ];\n * \n * // En el template\n * <pdm-data-table\n *   [columns]=\"columns\"\n *   [rows]=\"users\"\n *   [selectable]=\"true\"\n *   (selectionChange)=\"onSelect($event)\">\n * </pdm-data-table>\n */\n@Component({\n  selector: 'pdm-data-table',\n  templateUrl: './data-table.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PdmDataTableComponent<T = any> {\n  @Input() className = '';\n  \n  /**\n   * Columnas a mostrar\n   * Si no se provee, intenta inferir del primer row (legacy mode)\n   */\n  @Input() columns: PdmDataTableColumn<T>[] = [];\n  \n  /**\n   * Estrategia responsive de la tabla\n   */\n  @Input() responsiveStrategy: TableResponsiveStrategy = 'scroll';\n  \n  /**\n   * Si es true, muestra checkbox de selección en cada fila\n   */\n  @Input() selectable = false;\n  \n  /**\n   * Si es true, muestra botón de acciones (tres puntos) en cada fila\n   */\n  @Input() showActions = false;\n  \n  /**\n   * Si es true, muestra filtro de búsqueda\n   */\n  @Input() showFilter = true;\n  \n  /**\n   * Si es true, muestra controles de paginación\n   */\n  @Input() showPagination = true;\n  \n  /**\n   * Si es true, muestra selector de columnas\n   */\n  @Input() showColumnSelector = false;\n  \n  // Labels i18n\n  @Input() filterPlaceholder = 'Filter...';\n  @Input() columnsLabel = 'Columns';\n  @Input() previousLabel = 'Previous';\n  @Input() nextLabel = 'Next';\n  @Input() emptyLabel = 'No results.';\n  @Input() rowsSelectedLabel = 'row(s) selected';\n  \n  // DEPRECATED: Labels hardcodeados para backward compatibility\n  /**\n   * @deprecated Use columns configuration instead\n   */\n  @Input() statusLabel = 'Status';\n  /**\n   * @deprecated Use columns configuration instead\n   */\n  @Input() emailLabel = 'Email';\n  /**\n   * @deprecated Use columns configuration instead\n   */\n  @Input() amountLabel = 'Amount';\n  \n  /**\n   * Datos a mostrar\n   */\n  @Input() rows: T[] = [];\n  \n  /**\n   * Página actual (1-indexed)\n   */\n  @Input() page = 1;\n  \n  /**\n   * Cantidad de filas por página\n   */\n  @Input() pageSize = 10;\n  \n  /**\n   * Query de filtrado\n   */\n  @Input() query = '';\n  \n  /**\n   * Función custom de filtrado\n   * Si no se provee, busca en todos los campos string\n   */\n  @Input() filterFn?: (row: T, query: string) => boolean;\n\n  @Output() queryChange = new EventEmitter<string>();\n  @Output() rowAction = new EventEmitter<T>();\n  @Output() pageChange = new EventEmitter<number>();\n  @Output() selectionChange = new EventEmitter<{ row: T; selected: boolean }>();\n  @Output() columnSort = new EventEmitter<{ column: PdmDataTableColumn<T>; direction: 'asc' | 'desc' }>();\n\n  // Estado interno\n  selectedRows = new Set<T>();\n  sortColumn?: PdmDataTableColumn<T>;\n  sortDirection: 'asc' | 'desc' = 'asc';\n\n  /**\n   * Backward compatibility: si no hay columnas definidas, inferir del primer row\n   */\n  get effectiveColumns(): PdmDataTableColumn<T>[] {\n    if (this.columns.length > 0) {\n      return this.columns;\n    }\n    \n    // Legacy mode: inferir columnas del primer row (solo para PdmDataTableRow)\n    if (this.rows.length > 0) {\n      const firstRow = this.rows[0] as any;\n      return Object.keys(firstRow)\n        .filter(key => key !== 'selected')\n        .map(key => ({\n          key: key as keyof T,\n          label: this.getLegacyLabel(key),\n          align: key === 'amount' ? 'right' : 'left'\n        }));\n    }\n    \n    return [];\n  }\n\n  /**\n   * LEGACY: mapeo de keys a labels hardcodeados\n   */\n  private getLegacyLabel(key: string): string {\n    const map: Record<string, string> = {\n      status: this.statusLabel,\n      email: this.emailLabel,\n      amount: this.amountLabel\n    };\n    return map[key] || key.charAt(0).toUpperCase() + key.slice(1);\n  }\n\n  get filteredRows(): T[] {\n    const q = this.query.trim().toLowerCase();\n    if (!q) return this.rows;\n    \n    if (this.filterFn) {\n      return this.rows.filter(row => this.filterFn!(row, q));\n    }\n    \n    // Filtrado default: buscar en todos los campos string\n    return this.rows.filter(row => {\n      return Object.values(row as any).some(val => \n        typeof val === 'string' && val.toLowerCase().includes(q)\n      );\n    });\n  }\n\n  get pagedRows(): T[] {\n    const start = (this.page - 1) * this.pageSize;\n    return this.filteredRows.slice(start, start + this.pageSize);\n  }\n\n  get totalPages(): number {\n    return Math.max(1, Math.ceil(this.filteredRows.length / this.pageSize));\n  }\n\n  get selectedCount(): number {\n    return this.selectedRows.size;\n  }\n\n  onQueryInput(event: Event): void {\n    const value = (event.target as HTMLInputElement).value;\n    this.queryChange.emit(value);\n  }\n\n  onToggleRow(row: T, event: Event): void {\n    const checked = (event.target as HTMLInputElement).checked;\n    \n    if (checked) {\n      this.selectedRows.add(row);\n    } else {\n      this.selectedRows.delete(row);\n    }\n    \n    this.selectionChange.emit({ row, selected: checked });\n  }\n\n  onToggleAll(event: Event): void {\n    const checked = (event.target as HTMLInputElement).checked;\n    \n    if (checked) {\n      this.pagedRows.forEach(row => this.selectedRows.add(row));\n    } else {\n      this.pagedRows.forEach(row => this.selectedRows.delete(row));\n    }\n  }\n\n  isSelected(row: T): boolean {\n    return this.selectedRows.has(row);\n  }\n\n  previous(): void {\n    if (this.page <= 1) return;\n    this.pageChange.emit(this.page - 1);\n  }\n\n  next(): void {\n    if (this.page >= this.totalPages) return;\n    this.pageChange.emit(this.page + 1);\n  }\n\n  onAction(row: T): void {\n    this.rowAction.emit(row);\n  }\n\n  onSort(column: PdmDataTableColumn<T>): void {\n    if (!column.sortable) return;\n    \n    if (this.sortColumn === column) {\n      this.sortDirection = this.sortDirection === 'asc' ? 'desc' : 'asc';\n    } else {\n      this.sortColumn = column;\n      this.sortDirection = 'asc';\n    }\n    \n    this.columnSort.emit({ column, direction: this.sortDirection });\n  }\n\n  getCellValue(row: T, column: PdmDataTableColumn<T>): any {\n    const value = row[column.key];\n    \n    if (column.render) {\n      return column.render(value, row);\n    }\n    \n    return value != null ? String(value) : '';\n  }\n\n  getCellClass(column: PdmDataTableColumn<T>): string {\n    const classes = ['px-2', 'py-2'];\n    \n    if (column.align === 'center') classes.push('text-center');\n    if (column.align === 'right') classes.push('text-right');\n    if (column.hideOnMobile) classes.push('hidden', 'md:table-cell');\n    if (column.cellClass) classes.push(column.cellClass);\n    \n    return classes.join(' ');\n  }\n\n  getHeaderClass(column: PdmDataTableColumn<T>): string {\n    const classes = ['px-2', 'py-2', 'text-left', 'font-medium'];\n    \n    if (column.hideOnMobile) classes.push('hidden', 'md:table-cell');\n    if (column.headerClass) classes.push(column.headerClass);\n    \n    return classes.join(' ');\n  }\n\n  getColumnStyle(column: PdmDataTableColumn<T>): any {\n    return column.width ? { width: column.width } : {};\n  }\n}\n","<section [ngClass]=\"['flex w-full flex-col', className]\">\n  <!-- Toolbar: Filtro + Selector de columnas -->\n  <div *ngIf=\"showFilter || showColumnSelector\" class=\"flex w-full items-center justify-between gap-2 py-4\">\n    <input\n      *ngIf=\"showFilter\"\n      type=\"text\"\n      [placeholder]=\"filterPlaceholder\"\n      [value]=\"query\"\n      (input)=\"onQueryInput($event)\"\n      class=\"h-9 flex-1 rounded-md border border-input bg-transparent px-3 py-1 text-sm text-foreground shadow-sm placeholder:text-muted-foreground outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background\"\n    />\n\n    <button \n      *ngIf=\"showColumnSelector\"\n      type=\"button\" \n      class=\"inline-flex h-9 appearance-none items-center gap-2 rounded-md border border-input bg-background px-3 py-2 text-sm font-medium text-foreground shadow-sm whitespace-nowrap\">\n      <span>{{ columnsLabel }}</span>\n      <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4 text-foreground\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n        <path d=\"M7 10L12 15L17 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n      </svg>\n    </button>\n  </div>\n\n  <!-- Tabla con responsive -->\n  <pdm-table \n    variant=\"data\"\n    [responsiveStrategy]=\"responsiveStrategy\"\n    [fullBleed]=\"false\">\n    <thead>\n      <tr>\n        <!-- Columna de selección -->\n        <th *ngIf=\"selectable\" class=\"w-10 px-2 py-2 text-left font-medium\">\n          <input \n            type=\"checkbox\" \n            (change)=\"onToggleAll($event)\"\n            class=\"h-4 w-4 rounded-sm border border-input\" \n          />\n        </th>\n\n        <!-- Columnas dinámicas -->\n        <th \n          *ngFor=\"let column of effectiveColumns\"\n          [ngClass]=\"getHeaderClass(column)\"\n          [ngStyle]=\"getColumnStyle(column)\">\n          \n          <!-- Header sortable -->\n          <button \n            *ngIf=\"column.sortable\"\n            type=\"button\" \n            (click)=\"onSort(column)\"\n            class=\"inline-flex appearance-none items-center gap-1 rounded-sm border-0 bg-transparent px-3 py-2 text-sm hover:underline\">\n            <span>{{ column.label }}</span>\n            <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n              <path d=\"M8 6L4 10L8 14M16 18L20 14L16 10\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\" stroke-linejoin=\"round\"></path>\n            </svg>\n          </button>\n\n          <!-- Header no sortable -->\n          <span *ngIf=\"!column.sortable\">{{ column.label }}</span>\n        </th>\n\n        <!-- Columna de acciones -->\n        <th *ngIf=\"showActions\" class=\"w-10 px-2 py-2\"></th>\n      </tr>\n    </thead>\n\n    <tbody>\n      <!-- Filas con datos -->\n      <tr *ngFor=\"let row of pagedRows\">\n        <!-- Celda de selección -->\n        <td *ngIf=\"selectable\" class=\"px-2 py-2\">\n          <input \n            type=\"checkbox\" \n            [checked]=\"isSelected(row)\" \n            (change)=\"onToggleRow(row, $event)\" \n            class=\"h-4 w-4 rounded-sm border border-input\" \n          />\n        </td>\n\n        <!-- Celdas dinámicas -->\n        <td \n          *ngFor=\"let column of effectiveColumns\"\n          [ngClass]=\"getCellClass(column)\">\n          \n          <!-- Template personalizado si existe -->\n          <ng-container *ngIf=\"column.cellTemplate; else defaultCell\">\n            <ng-container \n              *ngTemplateOutlet=\"column.cellTemplate; context: { $implicit: row, value: row[column.key] }\">\n            </ng-container>\n          </ng-container>\n\n          <!-- Renderizado default -->\n          <ng-template #defaultCell>\n            {{ getCellValue(row, column) }}\n          </ng-template>\n        </td>\n\n        <!-- Celda de acciones -->\n        <td *ngIf=\"showActions\" class=\"px-2 py-2\">\n          <button \n            type=\"button\" \n            class=\"inline-flex h-8 w-8 appearance-none items-center justify-center border-0 bg-transparent p-0 hover:text-foreground\" \n            (click)=\"onAction(row)\">\n            <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n              <circle cx=\"6\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n              <circle cx=\"12\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n              <circle cx=\"18\" cy=\"12\" r=\"1.5\" fill=\"currentColor\"></circle>\n            </svg>\n          </button>\n        </td>\n      </tr>\n\n      <!-- Fila vacía -->\n      <tr *ngIf=\"pagedRows.length === 0\">\n        <td \n          [attr.colspan]=\"effectiveColumns.length + (selectable ? 1 : 0) + (showActions ? 1 : 0)\" \n          class=\"px-3 py-6 text-center text-sm text-muted-foreground\">\n          {{ emptyLabel }}\n        </td>\n      </tr>\n    </tbody>\n  </pdm-table>\n\n  <!-- Footer: Info + Paginación -->\n  <div *ngIf=\"showPagination || selectable\" class=\"flex w-full items-center gap-2 py-4 flex-wrap sm:flex-nowrap\">\n    <p *ngIf=\"selectable\" class=\"m-0 flex-1 pr-2 text-sm text-muted-foreground whitespace-nowrap\">\n      {{ selectedCount }} of {{ rows.length }} {{ rowsSelectedLabel }}\n    </p>\n\n    <div *ngIf=\"showPagination\" class=\"flex items-center gap-2 ml-auto\">\n      <span class=\"text-sm text-muted-foreground whitespace-nowrap\">\n        Page {{ page }} of {{ totalPages }}\n      </span>\n      <button \n        type=\"button\" \n        class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:cursor-not-allowed\" \n        [disabled]=\"page <= 1\" \n        (click)=\"previous()\">\n        {{ previousLabel }}\n      </button>\n      <button \n        type=\"button\" \n        class=\"h-9 appearance-none rounded-md border border-input bg-background px-4 text-sm font-medium text-foreground shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:cursor-not-allowed\" \n        [disabled]=\"page >= totalPages\" \n        (click)=\"next()\">\n        {{ nextLabel }}\n      </button>\n    </div>\n  </div>\n</section>\n"]}