pdm-ui-kit 0.1.50 → 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.
- package/README.md +189 -2
- package/esm2020/lib/components/alert-dialog/alert-dialog.component.mjs +3 -3
- package/esm2020/lib/components/breadcrumb/breadcrumb.component.mjs +37 -4
- package/esm2020/lib/components/calendar/calendar.component.mjs +3 -3
- package/esm2020/lib/components/card/card.component.mjs +36 -53
- package/esm2020/lib/components/command/command.component.mjs +3 -3
- package/esm2020/lib/components/context-menu/context-menu.component.mjs +3 -3
- package/esm2020/lib/components/data-table/data-table.component.mjs +214 -16
- package/esm2020/lib/components/dialog/dialog.component.mjs +133 -17
- package/esm2020/lib/components/draggable-table/draggable-table.component.mjs +300 -0
- package/esm2020/lib/components/drawer/drawer.component.mjs +123 -16
- package/esm2020/lib/components/dropdown-menu/dropdown-menu.component.mjs +3 -2
- package/esm2020/lib/components/hover-card/hover-card.component.mjs +3 -3
- package/esm2020/lib/components/menubar/menubar.component.mjs +3 -3
- package/esm2020/lib/components/navigation-menu/navigation-menu.component.mjs +25 -3
- package/esm2020/lib/components/pagination/pagination.component.mjs +3 -3
- package/esm2020/lib/components/popover/popover.component.mjs +5 -3
- package/esm2020/lib/components/select/select.component.mjs +5 -3
- package/esm2020/lib/components/sheet/sheet.component.mjs +68 -12
- package/esm2020/lib/components/sidebar/sidebar.component.mjs +52 -5
- package/esm2020/lib/components/table/table.component.mjs +152 -188
- package/esm2020/lib/components/tabs/tabs.component.mjs +3 -3
- package/esm2020/lib/components/tooltip/tooltip.component.mjs +3 -3
- package/esm2020/lib/pdm-ui-kit.module.mjs +5 -1
- package/esm2020/lib/utils/responsive.mjs +143 -0
- package/esm2020/lib/utils/z-index.mjs +93 -0
- package/esm2020/public-api.mjs +4 -1
- package/fesm2015/pdm-ui-kit.mjs +1430 -371
- package/fesm2015/pdm-ui-kit.mjs.map +1 -1
- package/fesm2020/pdm-ui-kit.mjs +1428 -369
- package/fesm2020/pdm-ui-kit.mjs.map +1 -1
- package/lib/components/breadcrumb/breadcrumb.component.d.ts +23 -1
- package/lib/components/card/card.component.d.ts +32 -19
- package/lib/components/data-table/data-table.component.d.ts +172 -14
- package/lib/components/dialog/dialog.component.d.ts +35 -1
- package/lib/components/draggable-table/draggable-table.component.d.ts +74 -0
- package/lib/components/drawer/drawer.component.d.ts +65 -7
- package/lib/components/navigation-menu/navigation-menu.component.d.ts +22 -1
- package/lib/components/sheet/sheet.component.d.ts +30 -3
- package/lib/components/sidebar/sidebar.component.d.ts +39 -1
- package/lib/components/table/table.component.d.ts +46 -25
- package/lib/pdm-ui-kit.module.d.ts +42 -41
- package/lib/utils/responsive.d.ts +107 -0
- package/lib/utils/z-index.d.ts +73 -0
- package/package.json +5 -3
- package/public-api.d.ts +3 -0
|
@@ -63,10 +63,10 @@ export class PdmContextMenuComponent {
|
|
|
63
63
|
}
|
|
64
64
|
}
|
|
65
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 });
|
|
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-
|
|
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 });
|
|
67
67
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmContextMenuComponent, decorators: [{
|
|
68
68
|
type: Component,
|
|
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-
|
|
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" }]
|
|
70
70
|
}], ctorParameters: function () { return [{ type: i0.ElementRef }]; }, propDecorators: { items: [{
|
|
71
71
|
type: Input
|
|
72
72
|
}], className: [{
|
|
@@ -83,4 +83,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
83
83
|
type: HostListener,
|
|
84
84
|
args: ['document:keydown.escape']
|
|
85
85
|
}] } });
|
|
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-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
|
-
|
|
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
|
-
|
|
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.
|
|
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
|
-
|
|
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
|
|
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",
|
|
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
|
|
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
|
-
}],
|
|
246
|
+
}], columns: [{
|
|
69
247
|
type: Input
|
|
70
|
-
}],
|
|
248
|
+
}], responsiveStrategy: [{
|
|
71
249
|
type: Input
|
|
72
|
-
}],
|
|
250
|
+
}], selectable: [{
|
|
73
251
|
type: Input
|
|
74
|
-
}],
|
|
252
|
+
}], showActions: [{
|
|
75
253
|
type: Input
|
|
76
|
-
}],
|
|
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"]}
|