valtech-components 2.0.583 → 2.0.584

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.
@@ -320,9 +320,16 @@ export class DataTableComponent {
320
320
  this.emitSelectionChange(checked ? 'select' : 'deselect', row);
321
321
  }
322
322
  selectSingleRow(row) {
323
+ const rowId = this.getRowId(row);
324
+ const wasSelected = this.selectedRows.has(rowId);
323
325
  this.selectedRows.clear();
324
- this.selectedRows.add(this.getRowId(row));
325
- this.emitSelectionChange('select', row);
326
+ if (!wasSelected) {
327
+ this.selectedRows.add(rowId);
328
+ this.emitSelectionChange('select', row);
329
+ }
330
+ else {
331
+ this.emitSelectionChange('deselect', row);
332
+ }
326
333
  }
327
334
  emitSelectionChange(action, changedRow) {
328
335
  const selected = this.props.data.filter(row => this.isRowSelected(row));
@@ -518,8 +525,7 @@ export class DataTableComponent {
518
525
  <input
519
526
  type="radio"
520
527
  [checked]="isRowSelected(row)"
521
- (change)="selectSingleRow(row)"
522
- (click)="$event.stopPropagation()"
528
+ (click)="selectSingleRow(row); $event.stopPropagation()"
523
529
  />
524
530
  }
525
531
  </td>
@@ -597,8 +603,7 @@ export class DataTableComponent {
597
603
  <input
598
604
  type="radio"
599
605
  [checked]="isRowSelected(row)"
600
- (change)="selectSingleRow(row)"
601
- (click)="$event.stopPropagation()"
606
+ (click)="selectSingleRow(row); $event.stopPropagation()"
602
607
  />
603
608
  }
604
609
  </div>
@@ -872,8 +877,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
872
877
  <input
873
878
  type="radio"
874
879
  [checked]="isRowSelected(row)"
875
- (change)="selectSingleRow(row)"
876
- (click)="$event.stopPropagation()"
880
+ (click)="selectSingleRow(row); $event.stopPropagation()"
877
881
  />
878
882
  }
879
883
  </td>
@@ -951,8 +955,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
951
955
  <input
952
956
  type="radio"
953
957
  [checked]="isRowSelected(row)"
954
- (change)="selectSingleRow(row)"
955
- (click)="$event.stopPropagation()"
958
+ (click)="selectSingleRow(row); $event.stopPropagation()"
956
959
  />
957
960
  }
958
961
  </div>
@@ -1075,4 +1078,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
1075
1078
  }], pageChange: [{
1076
1079
  type: Output
1077
1080
  }] } });
1078
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-table.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/data-table/data-table.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  OnInit,\n  OnChanges,\n  SimpleChanges,\n  inject,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport {\n  IonButton,\n  IonIcon,\n  IonCheckbox,\n  IonSpinner,\n  IonSelect,\n  IonSelectOption,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport {\n  chevronUpOutline,\n  chevronDownOutline,\n  chevronBackOutline,\n  chevronForwardOutline,\n  documentOutline,\n} from 'ionicons/icons';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  DataTableMetadata,\n  DataTableColumn,\n  DataTableSort,\n  DataTableRowClickEvent,\n  DataTableSelectionChangeEvent,\n  DataTableSortChangeEvent,\n  DataTablePageChangeEvent,\n  DataTableEmptyState,\n  DEFAULT_PAGE_SIZE_OPTIONS,\n} from './types';\n\naddIcons({\n  chevronUpOutline,\n  chevronDownOutline,\n  chevronBackOutline,\n  chevronForwardOutline,\n  documentOutline,\n});\n\n@Component({\n  selector: 'val-data-table',\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [\n    CommonModule,\n    FormsModule,\n    IonButton,\n    IonIcon,\n    IonCheckbox,\n    IonSpinner,\n    IonSelect,\n    IonSelectOption,\n  ],\n  template: `\n    <div\n      class=\"data-table-container\"\n      [class]=\"props.cssClass\"\n      [class.size-small]=\"props.size === 'small'\"\n      [class.size-medium]=\"props.size === 'medium' || !props.size\"\n      [class.size-large]=\"props.size === 'large'\"\n      [class.bordered]=\"props.bordered\"\n      [class.striped]=\"props.striped\"\n      [class.hoverable]=\"props.hoverable !== false\"\n      [class.sticky-header]=\"props.stickyHeader\"\n      [class.responsive-cards]=\"props.responsiveMode === 'cards'\"\n      [class.elevation-none]=\"props.elevation === 'none'\"\n      [class.elevation-low]=\"props.elevation === 'low'\"\n      [class.elevation-high]=\"props.elevation === 'high'\"\n      [class.header-gradient]=\"props.headerGradient\"\n      [class.checkbox-circular]=\"props.checkboxStyle === 'circular'\"\n      [class.row-highlight-border]=\"props.rowHighlightStyle === 'border-left'\"\n      [class.row-highlight-both]=\"props.rowHighlightStyle === 'both'\"\n      [style.--max-height]=\"props.maxHeight\"\n    >\n      <!-- Toolbar slot -->\n      <div class=\"table-toolbar\" #toolbarContainer>\n        <ng-content select=\"[toolbar]\"></ng-content>\n      </div>\n\n      <!-- Loading overlay -->\n      @if (props.loadingState?.loading) {\n        <div class=\"loading-overlay\">\n          <ion-spinner name=\"crescent\"></ion-spinner>\n          @if (props.loadingState?.message) {\n            <span class=\"loading-message\">{{ props.loadingState.message }}</span>\n          }\n        </div>\n      }\n\n      <div class=\"table-wrapper\" [class.is-loading]=\"props.loadingState?.loading\">\n        <table\n          role=\"grid\"\n          [attr.aria-label]=\"props.ariaLabel\"\n        >\n          @if (props.caption) {\n            <caption class=\"sr-only\">{{ props.caption }}</caption>\n          }\n\n          <thead>\n            <tr>\n              <!-- Selection column -->\n              @if (props.showSelectionColumn && props.selection?.mode === 'multiple') {\n                <th class=\"selection-cell\">\n                  <ion-checkbox\n                    [checked]=\"isAllSelected\"\n                    [indeterminate]=\"isIndeterminate\"\n                    (ionChange)=\"toggleSelectAll($event)\"\n                  ></ion-checkbox>\n                </th>\n              }\n              @if (props.showSelectionColumn && props.selection?.mode === 'single') {\n                <th class=\"selection-cell\"></th>\n              }\n\n              <!-- Row number column -->\n              @if (props.showRowNumbers) {\n                <th class=\"row-number-cell\">#</th>\n              }\n\n              <!-- Data columns -->\n              @for (column of visibleColumns; track column.key) {\n                <th\n                  [class]=\"getColumnClass(column)\"\n                  [style.width]=\"column.width\"\n                  [style.min-width]=\"column.minWidth\"\n                  [style.max-width]=\"column.maxWidth\"\n                  [class.sortable]=\"column.sortable\"\n                  [class.sorted]=\"currentSort?.column === column.key\"\n                  (click)=\"column.sortable ? onSort(column) : null\"\n                  [attr.aria-sort]=\"getAriaSort(column)\"\n                >\n                  @if (column.headerTemplate) {\n                    <ng-container *ngTemplateOutlet=\"column.headerTemplate; context: { column }\"></ng-container>\n                  } @else {\n                    <span class=\"header-content\">\n                      {{ column.label }}\n                      @if (column.sortable) {\n                        <span class=\"sort-icons\">\n                          <ion-icon\n                            name=\"chevron-up-outline\"\n                            [class.active]=\"currentSort?.column === column.key && currentSort?.direction === 'asc'\"\n                          ></ion-icon>\n                          <ion-icon\n                            name=\"chevron-down-outline\"\n                            [class.active]=\"currentSort?.column === column.key && currentSort?.direction === 'desc'\"\n                          ></ion-icon>\n                        </span>\n                      }\n                    </span>\n                  }\n                </th>\n              }\n\n              <!-- Actions column -->\n              @if (props.actionsTemplate) {\n                <th class=\"actions-cell\" [style.width]=\"props.actionsWidth || '100px'\">\n                  {{ getActionsLabel() }}\n                </th>\n              }\n            </tr>\n          </thead>\n\n          <tbody>\n            @if (displayedData.length === 0 && !props.loadingState?.loading) {\n              <tr class=\"empty-row\">\n                <td [attr.colspan]=\"totalColumns\">\n                  <div class=\"empty-state\">\n                    @if (emptyState.template) {\n                      <ng-container *ngTemplateOutlet=\"emptyState.template\"></ng-container>\n                    } @else {\n                      @if (emptyState.icon) {\n                        <ion-icon [name]=\"emptyState.icon\" class=\"empty-icon\"></ion-icon>\n                      }\n                      @if (emptyState.title) {\n                        <h4 class=\"empty-title\">{{ emptyState.title }}</h4>\n                      }\n                      @if (emptyState.description) {\n                        <p class=\"empty-description\">{{ emptyState.description }}</p>\n                      }\n                    }\n                  </div>\n                </td>\n              </tr>\n            } @else {\n              @for (row of displayedData; track trackByFn(row); let i = $index) {\n                <tr\n                  [class.selected]=\"isRowSelected(row)\"\n                  [class.clickable]=\"props.rowClickable\"\n                  (click)=\"onRowClick(row, i, $event)\"\n                >\n                  <!-- Selection cell -->\n                  @if (props.showSelectionColumn) {\n                    <td class=\"selection-cell\">\n                      @if (props.selection?.mode === 'multiple') {\n                        <ion-checkbox\n                          [checked]=\"isRowSelected(row)\"\n                          (ionChange)=\"toggleRowSelection(row, $event)\"\n                          (click)=\"$event.stopPropagation()\"\n                        ></ion-checkbox>\n                      } @else if (props.selection?.mode === 'single') {\n                        <input\n                          type=\"radio\"\n                          [checked]=\"isRowSelected(row)\"\n                          (change)=\"selectSingleRow(row)\"\n                          (click)=\"$event.stopPropagation()\"\n                        />\n                      }\n                    </td>\n                  }\n\n                  <!-- Row number -->\n                  @if (props.showRowNumbers) {\n                    <td class=\"row-number-cell\">{{ getRowNumber(i) }}</td>\n                  }\n\n                  <!-- Data cells -->\n                  @for (column of visibleColumns; track column.key) {\n                    <td\n                      [class]=\"getColumnClass(column)\"\n                      [style.text-align]=\"column.align\"\n                    >\n                      @if (column.cellTemplate) {\n                        <ng-container *ngTemplateOutlet=\"column.cellTemplate; context: { row, column, value: getCellValue(row, column) }\"></ng-container>\n                      } @else {\n                        {{ getFormattedValue(row, column) }}\n                      }\n                    </td>\n                  }\n\n                  <!-- Actions cell -->\n                  @if (props.actionsTemplate) {\n                    <td class=\"actions-cell\">\n                      <ng-container *ngTemplateOutlet=\"props.actionsTemplate; context: { row, index: i }\"></ng-container>\n                    </td>\n                  }\n                </tr>\n              }\n            }\n          </tbody>\n        </table>\n      </div>\n\n      <!-- Card view (responsive mobile) -->\n      @if (props.responsiveMode === 'cards') {\n        <div class=\"card-list\">\n          @if (displayedData.length === 0 && !props.loadingState?.loading) {\n            <div class=\"empty-state\">\n              @if (emptyState.icon) {\n                <ion-icon [name]=\"emptyState.icon\" class=\"empty-icon\"></ion-icon>\n              }\n              @if (emptyState.title) {\n                <h4 class=\"empty-title\">{{ emptyState.title }}</h4>\n              }\n              @if (emptyState.description) {\n                <p class=\"empty-description\">{{ emptyState.description }}</p>\n              }\n            </div>\n          } @else {\n            @for (row of displayedData; track trackByFn(row); let i = $index) {\n              @if (props.mobileCardTemplate) {\n                <ng-container *ngTemplateOutlet=\"props.mobileCardTemplate; context: { row, index: i, columns: visibleColumns }\"></ng-container>\n              } @else {\n                <div\n                  class=\"data-card\"\n                  [class.selected]=\"isRowSelected(row)\"\n                  [class.clickable]=\"props.rowClickable\"\n                  (click)=\"onRowClick(row, i, $event)\"\n                >\n                  <!-- Card header with selection -->\n                  @if (props.showSelectionColumn) {\n                    <div class=\"card-header\">\n                      <div class=\"card-selection\">\n                        @if (props.selection?.mode === 'multiple') {\n                          <ion-checkbox\n                            [checked]=\"isRowSelected(row)\"\n                            (ionChange)=\"toggleRowSelection(row, $event)\"\n                            (click)=\"$event.stopPropagation()\"\n                          ></ion-checkbox>\n                        } @else if (props.selection?.mode === 'single') {\n                          <input\n                            type=\"radio\"\n                            [checked]=\"isRowSelected(row)\"\n                            (change)=\"selectSingleRow(row)\"\n                            (click)=\"$event.stopPropagation()\"\n                          />\n                        }\n                      </div>\n                      @if (props.showRowNumbers) {\n                        <span class=\"row-number\">{{ getRowNumber(i) }}</span>\n                      }\n                    </div>\n                  }\n\n                  <!-- Card body with fields -->\n                  <div class=\"card-body\">\n                    @for (column of visibleColumns; track column.key) {\n                      <div class=\"card-field\">\n                        <span class=\"card-field__label\">{{ column.label }}</span>\n                        <span class=\"card-field__value\">\n                          @if (column.cellTemplate) {\n                            <ng-container *ngTemplateOutlet=\"column.cellTemplate; context: { row, column, value: getCellValue(row, column) }\"></ng-container>\n                          } @else {\n                            {{ getFormattedValue(row, column) }}\n                          }\n                        </span>\n                      </div>\n                    }\n                  </div>\n\n                  <!-- Card actions -->\n                  @if (props.actionsTemplate) {\n                    <div class=\"card-actions\">\n                      <ng-container *ngTemplateOutlet=\"props.actionsTemplate; context: { row, index: i }\"></ng-container>\n                    </div>\n                  }\n                </div>\n              }\n            }\n          }\n        </div>\n      }\n\n      <!-- Pagination -->\n      @if (props.showPagination && props.pagination) {\n        <div class=\"pagination-container\">\n          <div class=\"pagination-info\">\n            <span>\n              {{ getPaginationInfoText() }}\n            </span>\n\n            @if (pageSizeOptions.length > 1) {\n              <ion-select\n                [value]=\"props.pagination.pageSize\"\n                (ionChange)=\"onPageSizeChange($event)\"\n                interface=\"popover\"\n                class=\"page-size-select\"\n              >\n                @for (size of pageSizeOptions; track size) {\n                  <ion-select-option [value]=\"size\">{{ getPerPageText(size) }}</ion-select-option>\n                }\n              </ion-select>\n            }\n          </div>\n\n          <div class=\"pagination-controls\">\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              [disabled]=\"props.pagination.page === 0\"\n              (click)=\"goToPage(0)\"\n              [attr.aria-label]=\"getFirstPageLabel()\"\n            >\n              <ion-icon slot=\"icon-only\" name=\"chevron-back-outline\"></ion-icon>\n              <ion-icon slot=\"icon-only\" name=\"chevron-back-outline\" style=\"margin-left: -12px\"></ion-icon>\n            </ion-button>\n\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              [disabled]=\"props.pagination.page === 0\"\n              (click)=\"goToPage(props.pagination.page - 1)\"\n              [attr.aria-label]=\"getPreviousPageLabel()\"\n            >\n              <ion-icon slot=\"icon-only\" name=\"chevron-back-outline\"></ion-icon>\n            </ion-button>\n\n            <span class=\"page-indicator\">\n              {{ props.pagination.page + 1 }} / {{ totalPages }}\n            </span>\n\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              [disabled]=\"props.pagination.page >= totalPages - 1\"\n              (click)=\"goToPage(props.pagination.page + 1)\"\n              [attr.aria-label]=\"getNextPageLabel()\"\n            >\n              <ion-icon slot=\"icon-only\" name=\"chevron-forward-outline\"></ion-icon>\n            </ion-button>\n\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              [disabled]=\"props.pagination.page >= totalPages - 1\"\n              (click)=\"goToPage(totalPages - 1)\"\n              [attr.aria-label]=\"getLastPageLabel()\"\n            >\n              <ion-icon slot=\"icon-only\" name=\"chevron-forward-outline\"></ion-icon>\n              <ion-icon slot=\"icon-only\" name=\"chevron-forward-outline\" style=\"margin-left: -12px\"></ion-icon>\n            </ion-button>\n          </div>\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./data-table.component.scss'],\n})\n/**\n * val-data-table\n *\n * A flexible data table component for displaying tabular data with sorting,\n * pagination, and selection features.\n *\n * @example Basic usage\n * ```html\n * <val-data-table\n *   [props]=\"{\n *     columns: [\n *       { key: 'name', label: 'Nombre', sortable: true },\n *       { key: 'email', label: 'Email' },\n *       { key: 'status', label: 'Estado' }\n *     ],\n *     data: users\n *   }\"\n * ></val-data-table>\n * ```\n *\n * @example With pagination and selection\n * ```html\n * <val-data-table\n *   [props]=\"{\n *     columns: columns,\n *     data: participants,\n *     showPagination: true,\n *     pagination: {\n *       page: 0,\n *       pageSize: 25,\n *       total: totalCount\n *     },\n *     selection: {\n *       mode: 'multiple',\n *       selected: selectedParticipants\n *     },\n *     showSelectionColumn: true\n *   }\"\n *   (selectionChange)=\"onSelectionChange($event)\"\n *   (pageChange)=\"onPageChange($event)\"\n * ></val-data-table>\n * ```\n */\nexport class DataTableComponent<T = any> implements OnInit, OnChanges {\n  @Input() props: DataTableMetadata<T>;\n\n  @Output() rowClick = new EventEmitter<DataTableRowClickEvent<T>>();\n  @Output() selectionChange = new EventEmitter<DataTableSelectionChangeEvent<T>>();\n  @Output() sortChange = new EventEmitter<DataTableSortChangeEvent>();\n  @Output() pageChange = new EventEmitter<DataTablePageChangeEvent>();\n\n  displayedData: T[] = [];\n  currentSort: DataTableSort | null = null;\n  selectedRows: Set<any> = new Set();\n\n  /** Cached visible columns for performance */\n  private _visibleColumns: DataTableColumn<T>[] = [];\n\n  private cdr = inject(ChangeDetectorRef);\n  private i18n = inject(I18nService);\n\n  ngOnInit(): void {\n    this.initializeState();\n    this.updateDisplayedData();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['props']) {\n      this.initializeState();\n      this.updateDisplayedData();\n    }\n  }\n\n  private initializeState(): void {\n    // Cache visible columns for performance\n    this._visibleColumns = this.props.columns.filter(c => c.visible !== false);\n\n    // Initialize sort\n    this.currentSort = this.props.sort || null;\n\n    // Initialize selection\n    if (this.props.selection?.selected) {\n      this.selectedRows = new Set(\n        this.props.selection.selected.map(row => this.getRowId(row))\n      );\n    }\n  }\n\n  private updateDisplayedData(): void {\n    let data = [...this.props.data];\n\n    // Apply client-side sorting\n    if (this.props.clientSort && this.currentSort) {\n      data = this.sortData(data);\n    }\n\n    // Apply client-side pagination\n    if (this.props.clientPagination && this.props.pagination) {\n      const start = this.props.pagination.page * this.props.pagination.pageSize;\n      const end = start + this.props.pagination.pageSize;\n      data = data.slice(start, end);\n    }\n\n    this.displayedData = data;\n    this.cdr.markForCheck();\n  }\n\n  private sortData(data: T[]): T[] {\n    if (!this.currentSort) return data;\n\n    const column = this.props.columns.find(c => c.key === this.currentSort?.column);\n    if (!column) return data;\n\n    return [...data].sort((a, b) => {\n      if (column.sortFn) {\n        const result = column.sortFn(a, b);\n        return this.currentSort?.direction === 'desc' ? -result : result;\n      }\n\n      const valueA = this.getCellValue(a, column);\n      const valueB = this.getCellValue(b, column);\n\n      let comparison = 0;\n      if (valueA == null && valueB == null) comparison = 0;\n      else if (valueA == null) comparison = 1;\n      else if (valueB == null) comparison = -1;\n      else if (typeof valueA === 'string' && typeof valueB === 'string') {\n        comparison = valueA.localeCompare(valueB);\n      } else {\n        comparison = valueA < valueB ? -1 : valueA > valueB ? 1 : 0;\n      }\n\n      return this.currentSort?.direction === 'desc' ? -comparison : comparison;\n    });\n  }\n\n  get visibleColumns(): DataTableColumn<T>[] {\n    return this._visibleColumns;\n  }\n\n  get totalColumns(): number {\n    let count = this.visibleColumns.length;\n    if (this.props.showSelectionColumn) count++;\n    if (this.props.showRowNumbers) count++;\n    if (this.props.actionsTemplate) count++;\n    return count;\n  }\n\n  get emptyState(): DataTableEmptyState {\n    const defaultEmptyState: DataTableEmptyState = {\n      icon: 'document-outline',\n      title: this.i18n.t('noData'),\n      description: this.i18n.t('noRecordsFound'),\n    };\n    return this.props.emptyState || defaultEmptyState;\n  }\n\n  /** Get actions column label */\n  getActionsLabel(): string {\n    return this.props.actionsLabel || this.i18n.t('actions');\n  }\n\n  /** Get pagination info text */\n  getPaginationInfoText(): string {\n    return `${this.i18n.t('showing')} ${this.paginationStart}-${this.paginationEnd} ${this.i18n.t('of')} ${this.props.pagination?.total}`;\n  }\n\n  /** Get per page text */\n  getPerPageText(size: number): string {\n    return `${size} ${this.i18n.t('perPage')}`;\n  }\n\n  /** Get first page aria label */\n  getFirstPageLabel(): string {\n    return this.i18n.t('firstPage');\n  }\n\n  /** Get previous page aria label */\n  getPreviousPageLabel(): string {\n    return this.i18n.t('previousPage');\n  }\n\n  /** Get next page aria label */\n  getNextPageLabel(): string {\n    return this.i18n.t('nextPage');\n  }\n\n  /** Get last page aria label */\n  getLastPageLabel(): string {\n    return this.i18n.t('lastPage');\n  }\n\n  get pageSizeOptions(): number[] {\n    return this.props.pagination?.pageSizeOptions || DEFAULT_PAGE_SIZE_OPTIONS;\n  }\n\n  get totalPages(): number {\n    if (!this.props.pagination) return 1;\n    return Math.ceil(this.props.pagination.total / this.props.pagination.pageSize);\n  }\n\n  get paginationStart(): number {\n    if (!this.props.pagination) return 1;\n    return this.props.pagination.page * this.props.pagination.pageSize + 1;\n  }\n\n  get paginationEnd(): number {\n    if (!this.props.pagination) return this.displayedData.length;\n    return Math.min(\n      (this.props.pagination.page + 1) * this.props.pagination.pageSize,\n      this.props.pagination.total\n    );\n  }\n\n  get isAllSelected(): boolean {\n    if (this.displayedData.length === 0) return false;\n    return this.displayedData.every(row => this.isRowSelected(row));\n  }\n\n  get isIndeterminate(): boolean {\n    const selectedCount = this.displayedData.filter(row => this.isRowSelected(row)).length;\n    return selectedCount > 0 && selectedCount < this.displayedData.length;\n  }\n\n  getCellValue(row: T, column: DataTableColumn<T>): any {\n    const field = column.field || column.key;\n    return field.split('.').reduce((obj: any, key) => obj?.[key], row);\n  }\n\n  getFormattedValue(row: T, column: DataTableColumn<T>): string {\n    const value = this.getCellValue(row, column);\n    if (column.format) {\n      return column.format(value, row);\n    }\n    return value ?? '';\n  }\n\n  getColumnClass(column: DataTableColumn<T>): string {\n    const classes = [column.cssClass || ''];\n    if (column.sticky) {\n      classes.push(`sticky-${column.sticky}`);\n    }\n    if (column.align) {\n      classes.push(`align-${column.align}`);\n    }\n    return classes.filter(Boolean).join(' ');\n  }\n\n  getAriaSort(column: DataTableColumn<T>): string | null {\n    if (!column.sortable) return null;\n    if (this.currentSort?.column !== column.key) return 'none';\n    return this.currentSort.direction === 'asc' ? 'ascending' : 'descending';\n  }\n\n  getRowNumber(index: number): number {\n    const start = this.props.rowNumberStart ?? 1;\n    if (this.props.pagination) {\n      return start + this.props.pagination.page * this.props.pagination.pageSize + index;\n    }\n    return start + index;\n  }\n\n  getRowId(row: T): any {\n    if (this.props.selection?.trackBy) {\n      return this.props.selection.trackBy(row);\n    }\n    return row;\n  }\n\n  trackByFn(row: T): any {\n    return this.getRowId(row);\n  }\n\n  isRowSelected(row: T): boolean {\n    return this.selectedRows.has(this.getRowId(row));\n  }\n\n  onSort(column: DataTableColumn<T>): void {\n    if (!column.sortable) return;\n\n    let direction: 'asc' | 'desc' | null = 'asc';\n\n    if (this.currentSort?.column === column.key) {\n      if (this.currentSort.direction === 'asc') {\n        direction = 'desc';\n      } else {\n        direction = null;\n      }\n    }\n\n    if (direction) {\n      this.currentSort = { column: column.key, direction };\n    } else {\n      this.currentSort = null;\n    }\n\n    this.sortChange.emit({\n      column: column.key,\n      direction,\n    });\n\n    if (this.props.clientSort) {\n      this.updateDisplayedData();\n    }\n  }\n\n  onRowClick(row: T, index: number, event: MouseEvent): void {\n    if (!this.props.rowClickable) return;\n\n    this.rowClick.emit({\n      row,\n      index,\n      event,\n    });\n  }\n\n  toggleSelectAll(event: CustomEvent): void {\n    const checked = event.detail.checked;\n\n    if (checked) {\n      this.displayedData.forEach(row => {\n        this.selectedRows.add(this.getRowId(row));\n      });\n    } else {\n      this.displayedData.forEach(row => {\n        this.selectedRows.delete(this.getRowId(row));\n      });\n    }\n\n    this.emitSelectionChange(checked ? 'selectAll' : 'deselectAll');\n  }\n\n  toggleRowSelection(row: T, event: CustomEvent): void {\n    const rowId = this.getRowId(row);\n    const checked = event.detail.checked;\n\n    if (checked) {\n      this.selectedRows.add(rowId);\n    } else {\n      this.selectedRows.delete(rowId);\n    }\n\n    this.emitSelectionChange(checked ? 'select' : 'deselect', row);\n  }\n\n  selectSingleRow(row: T): void {\n    this.selectedRows.clear();\n    this.selectedRows.add(this.getRowId(row));\n    this.emitSelectionChange('select', row);\n  }\n\n  private emitSelectionChange(\n    action: 'select' | 'deselect' | 'selectAll' | 'deselectAll',\n    changedRow?: T\n  ): void {\n    const selected = this.props.data.filter(row => this.isRowSelected(row));\n\n    this.selectionChange.emit({\n      selected,\n      changedRow,\n      action,\n    });\n  }\n\n  goToPage(page: number): void {\n    if (!this.props.pagination) return;\n\n    const previousPage = this.props.pagination.page;\n    const maxPage = this.totalPages - 1;\n    const newPage = Math.max(0, Math.min(page, maxPage));\n\n    this.pageChange.emit({\n      page: newPage,\n      pageSize: this.props.pagination.pageSize,\n      previousPage,\n    });\n\n    if (this.props.clientPagination) {\n      this.props.pagination.page = newPage;\n      this.updateDisplayedData();\n    }\n  }\n\n  onPageSizeChange(event: CustomEvent): void {\n    if (!this.props.pagination) return;\n\n    const newPageSize = event.detail.value;\n    const previousPage = this.props.pagination.page;\n\n    // Recalculate page to keep first visible item\n    const firstItem = previousPage * this.props.pagination.pageSize;\n    const newPage = Math.floor(firstItem / newPageSize);\n\n    this.pageChange.emit({\n      page: newPage,\n      pageSize: newPageSize,\n      previousPage,\n    });\n\n    if (this.props.clientPagination) {\n      this.props.pagination.pageSize = newPageSize;\n      this.props.pagination.page = newPage;\n      this.updateDisplayedData();\n    }\n  }\n}\n"]}
1081
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-table.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/data-table/data-table.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  OnInit,\n  OnChanges,\n  SimpleChanges,\n  inject,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { FormsModule } from '@angular/forms';\nimport {\n  IonButton,\n  IonIcon,\n  IonCheckbox,\n  IonSpinner,\n  IonSelect,\n  IonSelectOption,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport {\n  chevronUpOutline,\n  chevronDownOutline,\n  chevronBackOutline,\n  chevronForwardOutline,\n  documentOutline,\n} from 'ionicons/icons';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  DataTableMetadata,\n  DataTableColumn,\n  DataTableSort,\n  DataTableRowClickEvent,\n  DataTableSelectionChangeEvent,\n  DataTableSortChangeEvent,\n  DataTablePageChangeEvent,\n  DataTableEmptyState,\n  DEFAULT_PAGE_SIZE_OPTIONS,\n} from './types';\n\naddIcons({\n  chevronUpOutline,\n  chevronDownOutline,\n  chevronBackOutline,\n  chevronForwardOutline,\n  documentOutline,\n});\n\n@Component({\n  selector: 'val-data-table',\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [\n    CommonModule,\n    FormsModule,\n    IonButton,\n    IonIcon,\n    IonCheckbox,\n    IonSpinner,\n    IonSelect,\n    IonSelectOption,\n  ],\n  template: `\n    <div\n      class=\"data-table-container\"\n      [class]=\"props.cssClass\"\n      [class.size-small]=\"props.size === 'small'\"\n      [class.size-medium]=\"props.size === 'medium' || !props.size\"\n      [class.size-large]=\"props.size === 'large'\"\n      [class.bordered]=\"props.bordered\"\n      [class.striped]=\"props.striped\"\n      [class.hoverable]=\"props.hoverable !== false\"\n      [class.sticky-header]=\"props.stickyHeader\"\n      [class.responsive-cards]=\"props.responsiveMode === 'cards'\"\n      [class.elevation-none]=\"props.elevation === 'none'\"\n      [class.elevation-low]=\"props.elevation === 'low'\"\n      [class.elevation-high]=\"props.elevation === 'high'\"\n      [class.header-gradient]=\"props.headerGradient\"\n      [class.checkbox-circular]=\"props.checkboxStyle === 'circular'\"\n      [class.row-highlight-border]=\"props.rowHighlightStyle === 'border-left'\"\n      [class.row-highlight-both]=\"props.rowHighlightStyle === 'both'\"\n      [style.--max-height]=\"props.maxHeight\"\n    >\n      <!-- Toolbar slot -->\n      <div class=\"table-toolbar\" #toolbarContainer>\n        <ng-content select=\"[toolbar]\"></ng-content>\n      </div>\n\n      <!-- Loading overlay -->\n      @if (props.loadingState?.loading) {\n        <div class=\"loading-overlay\">\n          <ion-spinner name=\"crescent\"></ion-spinner>\n          @if (props.loadingState?.message) {\n            <span class=\"loading-message\">{{ props.loadingState.message }}</span>\n          }\n        </div>\n      }\n\n      <div class=\"table-wrapper\" [class.is-loading]=\"props.loadingState?.loading\">\n        <table\n          role=\"grid\"\n          [attr.aria-label]=\"props.ariaLabel\"\n        >\n          @if (props.caption) {\n            <caption class=\"sr-only\">{{ props.caption }}</caption>\n          }\n\n          <thead>\n            <tr>\n              <!-- Selection column -->\n              @if (props.showSelectionColumn && props.selection?.mode === 'multiple') {\n                <th class=\"selection-cell\">\n                  <ion-checkbox\n                    [checked]=\"isAllSelected\"\n                    [indeterminate]=\"isIndeterminate\"\n                    (ionChange)=\"toggleSelectAll($event)\"\n                  ></ion-checkbox>\n                </th>\n              }\n              @if (props.showSelectionColumn && props.selection?.mode === 'single') {\n                <th class=\"selection-cell\"></th>\n              }\n\n              <!-- Row number column -->\n              @if (props.showRowNumbers) {\n                <th class=\"row-number-cell\">#</th>\n              }\n\n              <!-- Data columns -->\n              @for (column of visibleColumns; track column.key) {\n                <th\n                  [class]=\"getColumnClass(column)\"\n                  [style.width]=\"column.width\"\n                  [style.min-width]=\"column.minWidth\"\n                  [style.max-width]=\"column.maxWidth\"\n                  [class.sortable]=\"column.sortable\"\n                  [class.sorted]=\"currentSort?.column === column.key\"\n                  (click)=\"column.sortable ? onSort(column) : null\"\n                  [attr.aria-sort]=\"getAriaSort(column)\"\n                >\n                  @if (column.headerTemplate) {\n                    <ng-container *ngTemplateOutlet=\"column.headerTemplate; context: { column }\"></ng-container>\n                  } @else {\n                    <span class=\"header-content\">\n                      {{ column.label }}\n                      @if (column.sortable) {\n                        <span class=\"sort-icons\">\n                          <ion-icon\n                            name=\"chevron-up-outline\"\n                            [class.active]=\"currentSort?.column === column.key && currentSort?.direction === 'asc'\"\n                          ></ion-icon>\n                          <ion-icon\n                            name=\"chevron-down-outline\"\n                            [class.active]=\"currentSort?.column === column.key && currentSort?.direction === 'desc'\"\n                          ></ion-icon>\n                        </span>\n                      }\n                    </span>\n                  }\n                </th>\n              }\n\n              <!-- Actions column -->\n              @if (props.actionsTemplate) {\n                <th class=\"actions-cell\" [style.width]=\"props.actionsWidth || '100px'\">\n                  {{ getActionsLabel() }}\n                </th>\n              }\n            </tr>\n          </thead>\n\n          <tbody>\n            @if (displayedData.length === 0 && !props.loadingState?.loading) {\n              <tr class=\"empty-row\">\n                <td [attr.colspan]=\"totalColumns\">\n                  <div class=\"empty-state\">\n                    @if (emptyState.template) {\n                      <ng-container *ngTemplateOutlet=\"emptyState.template\"></ng-container>\n                    } @else {\n                      @if (emptyState.icon) {\n                        <ion-icon [name]=\"emptyState.icon\" class=\"empty-icon\"></ion-icon>\n                      }\n                      @if (emptyState.title) {\n                        <h4 class=\"empty-title\">{{ emptyState.title }}</h4>\n                      }\n                      @if (emptyState.description) {\n                        <p class=\"empty-description\">{{ emptyState.description }}</p>\n                      }\n                    }\n                  </div>\n                </td>\n              </tr>\n            } @else {\n              @for (row of displayedData; track trackByFn(row); let i = $index) {\n                <tr\n                  [class.selected]=\"isRowSelected(row)\"\n                  [class.clickable]=\"props.rowClickable\"\n                  (click)=\"onRowClick(row, i, $event)\"\n                >\n                  <!-- Selection cell -->\n                  @if (props.showSelectionColumn) {\n                    <td class=\"selection-cell\">\n                      @if (props.selection?.mode === 'multiple') {\n                        <ion-checkbox\n                          [checked]=\"isRowSelected(row)\"\n                          (ionChange)=\"toggleRowSelection(row, $event)\"\n                          (click)=\"$event.stopPropagation()\"\n                        ></ion-checkbox>\n                      } @else if (props.selection?.mode === 'single') {\n                        <input\n                          type=\"radio\"\n                          [checked]=\"isRowSelected(row)\"\n                          (click)=\"selectSingleRow(row); $event.stopPropagation()\"\n                        />\n                      }\n                    </td>\n                  }\n\n                  <!-- Row number -->\n                  @if (props.showRowNumbers) {\n                    <td class=\"row-number-cell\">{{ getRowNumber(i) }}</td>\n                  }\n\n                  <!-- Data cells -->\n                  @for (column of visibleColumns; track column.key) {\n                    <td\n                      [class]=\"getColumnClass(column)\"\n                      [style.text-align]=\"column.align\"\n                    >\n                      @if (column.cellTemplate) {\n                        <ng-container *ngTemplateOutlet=\"column.cellTemplate; context: { row, column, value: getCellValue(row, column) }\"></ng-container>\n                      } @else {\n                        {{ getFormattedValue(row, column) }}\n                      }\n                    </td>\n                  }\n\n                  <!-- Actions cell -->\n                  @if (props.actionsTemplate) {\n                    <td class=\"actions-cell\">\n                      <ng-container *ngTemplateOutlet=\"props.actionsTemplate; context: { row, index: i }\"></ng-container>\n                    </td>\n                  }\n                </tr>\n              }\n            }\n          </tbody>\n        </table>\n      </div>\n\n      <!-- Card view (responsive mobile) -->\n      @if (props.responsiveMode === 'cards') {\n        <div class=\"card-list\">\n          @if (displayedData.length === 0 && !props.loadingState?.loading) {\n            <div class=\"empty-state\">\n              @if (emptyState.icon) {\n                <ion-icon [name]=\"emptyState.icon\" class=\"empty-icon\"></ion-icon>\n              }\n              @if (emptyState.title) {\n                <h4 class=\"empty-title\">{{ emptyState.title }}</h4>\n              }\n              @if (emptyState.description) {\n                <p class=\"empty-description\">{{ emptyState.description }}</p>\n              }\n            </div>\n          } @else {\n            @for (row of displayedData; track trackByFn(row); let i = $index) {\n              @if (props.mobileCardTemplate) {\n                <ng-container *ngTemplateOutlet=\"props.mobileCardTemplate; context: { row, index: i, columns: visibleColumns }\"></ng-container>\n              } @else {\n                <div\n                  class=\"data-card\"\n                  [class.selected]=\"isRowSelected(row)\"\n                  [class.clickable]=\"props.rowClickable\"\n                  (click)=\"onRowClick(row, i, $event)\"\n                >\n                  <!-- Card header with selection -->\n                  @if (props.showSelectionColumn) {\n                    <div class=\"card-header\">\n                      <div class=\"card-selection\">\n                        @if (props.selection?.mode === 'multiple') {\n                          <ion-checkbox\n                            [checked]=\"isRowSelected(row)\"\n                            (ionChange)=\"toggleRowSelection(row, $event)\"\n                            (click)=\"$event.stopPropagation()\"\n                          ></ion-checkbox>\n                        } @else if (props.selection?.mode === 'single') {\n                          <input\n                            type=\"radio\"\n                            [checked]=\"isRowSelected(row)\"\n                            (click)=\"selectSingleRow(row); $event.stopPropagation()\"\n                          />\n                        }\n                      </div>\n                      @if (props.showRowNumbers) {\n                        <span class=\"row-number\">{{ getRowNumber(i) }}</span>\n                      }\n                    </div>\n                  }\n\n                  <!-- Card body with fields -->\n                  <div class=\"card-body\">\n                    @for (column of visibleColumns; track column.key) {\n                      <div class=\"card-field\">\n                        <span class=\"card-field__label\">{{ column.label }}</span>\n                        <span class=\"card-field__value\">\n                          @if (column.cellTemplate) {\n                            <ng-container *ngTemplateOutlet=\"column.cellTemplate; context: { row, column, value: getCellValue(row, column) }\"></ng-container>\n                          } @else {\n                            {{ getFormattedValue(row, column) }}\n                          }\n                        </span>\n                      </div>\n                    }\n                  </div>\n\n                  <!-- Card actions -->\n                  @if (props.actionsTemplate) {\n                    <div class=\"card-actions\">\n                      <ng-container *ngTemplateOutlet=\"props.actionsTemplate; context: { row, index: i }\"></ng-container>\n                    </div>\n                  }\n                </div>\n              }\n            }\n          }\n        </div>\n      }\n\n      <!-- Pagination -->\n      @if (props.showPagination && props.pagination) {\n        <div class=\"pagination-container\">\n          <div class=\"pagination-info\">\n            <span>\n              {{ getPaginationInfoText() }}\n            </span>\n\n            @if (pageSizeOptions.length > 1) {\n              <ion-select\n                [value]=\"props.pagination.pageSize\"\n                (ionChange)=\"onPageSizeChange($event)\"\n                interface=\"popover\"\n                class=\"page-size-select\"\n              >\n                @for (size of pageSizeOptions; track size) {\n                  <ion-select-option [value]=\"size\">{{ getPerPageText(size) }}</ion-select-option>\n                }\n              </ion-select>\n            }\n          </div>\n\n          <div class=\"pagination-controls\">\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              [disabled]=\"props.pagination.page === 0\"\n              (click)=\"goToPage(0)\"\n              [attr.aria-label]=\"getFirstPageLabel()\"\n            >\n              <ion-icon slot=\"icon-only\" name=\"chevron-back-outline\"></ion-icon>\n              <ion-icon slot=\"icon-only\" name=\"chevron-back-outline\" style=\"margin-left: -12px\"></ion-icon>\n            </ion-button>\n\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              [disabled]=\"props.pagination.page === 0\"\n              (click)=\"goToPage(props.pagination.page - 1)\"\n              [attr.aria-label]=\"getPreviousPageLabel()\"\n            >\n              <ion-icon slot=\"icon-only\" name=\"chevron-back-outline\"></ion-icon>\n            </ion-button>\n\n            <span class=\"page-indicator\">\n              {{ props.pagination.page + 1 }} / {{ totalPages }}\n            </span>\n\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              [disabled]=\"props.pagination.page >= totalPages - 1\"\n              (click)=\"goToPage(props.pagination.page + 1)\"\n              [attr.aria-label]=\"getNextPageLabel()\"\n            >\n              <ion-icon slot=\"icon-only\" name=\"chevron-forward-outline\"></ion-icon>\n            </ion-button>\n\n            <ion-button\n              fill=\"clear\"\n              size=\"small\"\n              [disabled]=\"props.pagination.page >= totalPages - 1\"\n              (click)=\"goToPage(totalPages - 1)\"\n              [attr.aria-label]=\"getLastPageLabel()\"\n            >\n              <ion-icon slot=\"icon-only\" name=\"chevron-forward-outline\"></ion-icon>\n              <ion-icon slot=\"icon-only\" name=\"chevron-forward-outline\" style=\"margin-left: -12px\"></ion-icon>\n            </ion-button>\n          </div>\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./data-table.component.scss'],\n})\n/**\n * val-data-table\n *\n * A flexible data table component for displaying tabular data with sorting,\n * pagination, and selection features.\n *\n * @example Basic usage\n * ```html\n * <val-data-table\n *   [props]=\"{\n *     columns: [\n *       { key: 'name', label: 'Nombre', sortable: true },\n *       { key: 'email', label: 'Email' },\n *       { key: 'status', label: 'Estado' }\n *     ],\n *     data: users\n *   }\"\n * ></val-data-table>\n * ```\n *\n * @example With pagination and selection\n * ```html\n * <val-data-table\n *   [props]=\"{\n *     columns: columns,\n *     data: participants,\n *     showPagination: true,\n *     pagination: {\n *       page: 0,\n *       pageSize: 25,\n *       total: totalCount\n *     },\n *     selection: {\n *       mode: 'multiple',\n *       selected: selectedParticipants\n *     },\n *     showSelectionColumn: true\n *   }\"\n *   (selectionChange)=\"onSelectionChange($event)\"\n *   (pageChange)=\"onPageChange($event)\"\n * ></val-data-table>\n * ```\n */\nexport class DataTableComponent<T = any> implements OnInit, OnChanges {\n  @Input() props: DataTableMetadata<T>;\n\n  @Output() rowClick = new EventEmitter<DataTableRowClickEvent<T>>();\n  @Output() selectionChange = new EventEmitter<DataTableSelectionChangeEvent<T>>();\n  @Output() sortChange = new EventEmitter<DataTableSortChangeEvent>();\n  @Output() pageChange = new EventEmitter<DataTablePageChangeEvent>();\n\n  displayedData: T[] = [];\n  currentSort: DataTableSort | null = null;\n  selectedRows: Set<any> = new Set();\n\n  /** Cached visible columns for performance */\n  private _visibleColumns: DataTableColumn<T>[] = [];\n\n  private cdr = inject(ChangeDetectorRef);\n  private i18n = inject(I18nService);\n\n  ngOnInit(): void {\n    this.initializeState();\n    this.updateDisplayedData();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['props']) {\n      this.initializeState();\n      this.updateDisplayedData();\n    }\n  }\n\n  private initializeState(): void {\n    // Cache visible columns for performance\n    this._visibleColumns = this.props.columns.filter(c => c.visible !== false);\n\n    // Initialize sort\n    this.currentSort = this.props.sort || null;\n\n    // Initialize selection\n    if (this.props.selection?.selected) {\n      this.selectedRows = new Set(\n        this.props.selection.selected.map(row => this.getRowId(row))\n      );\n    }\n  }\n\n  private updateDisplayedData(): void {\n    let data = [...this.props.data];\n\n    // Apply client-side sorting\n    if (this.props.clientSort && this.currentSort) {\n      data = this.sortData(data);\n    }\n\n    // Apply client-side pagination\n    if (this.props.clientPagination && this.props.pagination) {\n      const start = this.props.pagination.page * this.props.pagination.pageSize;\n      const end = start + this.props.pagination.pageSize;\n      data = data.slice(start, end);\n    }\n\n    this.displayedData = data;\n    this.cdr.markForCheck();\n  }\n\n  private sortData(data: T[]): T[] {\n    if (!this.currentSort) return data;\n\n    const column = this.props.columns.find(c => c.key === this.currentSort?.column);\n    if (!column) return data;\n\n    return [...data].sort((a, b) => {\n      if (column.sortFn) {\n        const result = column.sortFn(a, b);\n        return this.currentSort?.direction === 'desc' ? -result : result;\n      }\n\n      const valueA = this.getCellValue(a, column);\n      const valueB = this.getCellValue(b, column);\n\n      let comparison = 0;\n      if (valueA == null && valueB == null) comparison = 0;\n      else if (valueA == null) comparison = 1;\n      else if (valueB == null) comparison = -1;\n      else if (typeof valueA === 'string' && typeof valueB === 'string') {\n        comparison = valueA.localeCompare(valueB);\n      } else {\n        comparison = valueA < valueB ? -1 : valueA > valueB ? 1 : 0;\n      }\n\n      return this.currentSort?.direction === 'desc' ? -comparison : comparison;\n    });\n  }\n\n  get visibleColumns(): DataTableColumn<T>[] {\n    return this._visibleColumns;\n  }\n\n  get totalColumns(): number {\n    let count = this.visibleColumns.length;\n    if (this.props.showSelectionColumn) count++;\n    if (this.props.showRowNumbers) count++;\n    if (this.props.actionsTemplate) count++;\n    return count;\n  }\n\n  get emptyState(): DataTableEmptyState {\n    const defaultEmptyState: DataTableEmptyState = {\n      icon: 'document-outline',\n      title: this.i18n.t('noData'),\n      description: this.i18n.t('noRecordsFound'),\n    };\n    return this.props.emptyState || defaultEmptyState;\n  }\n\n  /** Get actions column label */\n  getActionsLabel(): string {\n    return this.props.actionsLabel || this.i18n.t('actions');\n  }\n\n  /** Get pagination info text */\n  getPaginationInfoText(): string {\n    return `${this.i18n.t('showing')} ${this.paginationStart}-${this.paginationEnd} ${this.i18n.t('of')} ${this.props.pagination?.total}`;\n  }\n\n  /** Get per page text */\n  getPerPageText(size: number): string {\n    return `${size} ${this.i18n.t('perPage')}`;\n  }\n\n  /** Get first page aria label */\n  getFirstPageLabel(): string {\n    return this.i18n.t('firstPage');\n  }\n\n  /** Get previous page aria label */\n  getPreviousPageLabel(): string {\n    return this.i18n.t('previousPage');\n  }\n\n  /** Get next page aria label */\n  getNextPageLabel(): string {\n    return this.i18n.t('nextPage');\n  }\n\n  /** Get last page aria label */\n  getLastPageLabel(): string {\n    return this.i18n.t('lastPage');\n  }\n\n  get pageSizeOptions(): number[] {\n    return this.props.pagination?.pageSizeOptions || DEFAULT_PAGE_SIZE_OPTIONS;\n  }\n\n  get totalPages(): number {\n    if (!this.props.pagination) return 1;\n    return Math.ceil(this.props.pagination.total / this.props.pagination.pageSize);\n  }\n\n  get paginationStart(): number {\n    if (!this.props.pagination) return 1;\n    return this.props.pagination.page * this.props.pagination.pageSize + 1;\n  }\n\n  get paginationEnd(): number {\n    if (!this.props.pagination) return this.displayedData.length;\n    return Math.min(\n      (this.props.pagination.page + 1) * this.props.pagination.pageSize,\n      this.props.pagination.total\n    );\n  }\n\n  get isAllSelected(): boolean {\n    if (this.displayedData.length === 0) return false;\n    return this.displayedData.every(row => this.isRowSelected(row));\n  }\n\n  get isIndeterminate(): boolean {\n    const selectedCount = this.displayedData.filter(row => this.isRowSelected(row)).length;\n    return selectedCount > 0 && selectedCount < this.displayedData.length;\n  }\n\n  getCellValue(row: T, column: DataTableColumn<T>): any {\n    const field = column.field || column.key;\n    return field.split('.').reduce((obj: any, key) => obj?.[key], row);\n  }\n\n  getFormattedValue(row: T, column: DataTableColumn<T>): string {\n    const value = this.getCellValue(row, column);\n    if (column.format) {\n      return column.format(value, row);\n    }\n    return value ?? '';\n  }\n\n  getColumnClass(column: DataTableColumn<T>): string {\n    const classes = [column.cssClass || ''];\n    if (column.sticky) {\n      classes.push(`sticky-${column.sticky}`);\n    }\n    if (column.align) {\n      classes.push(`align-${column.align}`);\n    }\n    return classes.filter(Boolean).join(' ');\n  }\n\n  getAriaSort(column: DataTableColumn<T>): string | null {\n    if (!column.sortable) return null;\n    if (this.currentSort?.column !== column.key) return 'none';\n    return this.currentSort.direction === 'asc' ? 'ascending' : 'descending';\n  }\n\n  getRowNumber(index: number): number {\n    const start = this.props.rowNumberStart ?? 1;\n    if (this.props.pagination) {\n      return start + this.props.pagination.page * this.props.pagination.pageSize + index;\n    }\n    return start + index;\n  }\n\n  getRowId(row: T): any {\n    if (this.props.selection?.trackBy) {\n      return this.props.selection.trackBy(row);\n    }\n    return row;\n  }\n\n  trackByFn(row: T): any {\n    return this.getRowId(row);\n  }\n\n  isRowSelected(row: T): boolean {\n    return this.selectedRows.has(this.getRowId(row));\n  }\n\n  onSort(column: DataTableColumn<T>): void {\n    if (!column.sortable) return;\n\n    let direction: 'asc' | 'desc' | null = 'asc';\n\n    if (this.currentSort?.column === column.key) {\n      if (this.currentSort.direction === 'asc') {\n        direction = 'desc';\n      } else {\n        direction = null;\n      }\n    }\n\n    if (direction) {\n      this.currentSort = { column: column.key, direction };\n    } else {\n      this.currentSort = null;\n    }\n\n    this.sortChange.emit({\n      column: column.key,\n      direction,\n    });\n\n    if (this.props.clientSort) {\n      this.updateDisplayedData();\n    }\n  }\n\n  onRowClick(row: T, index: number, event: MouseEvent): void {\n    if (!this.props.rowClickable) return;\n\n    this.rowClick.emit({\n      row,\n      index,\n      event,\n    });\n  }\n\n  toggleSelectAll(event: CustomEvent): void {\n    const checked = event.detail.checked;\n\n    if (checked) {\n      this.displayedData.forEach(row => {\n        this.selectedRows.add(this.getRowId(row));\n      });\n    } else {\n      this.displayedData.forEach(row => {\n        this.selectedRows.delete(this.getRowId(row));\n      });\n    }\n\n    this.emitSelectionChange(checked ? 'selectAll' : 'deselectAll');\n  }\n\n  toggleRowSelection(row: T, event: CustomEvent): void {\n    const rowId = this.getRowId(row);\n    const checked = event.detail.checked;\n\n    if (checked) {\n      this.selectedRows.add(rowId);\n    } else {\n      this.selectedRows.delete(rowId);\n    }\n\n    this.emitSelectionChange(checked ? 'select' : 'deselect', row);\n  }\n\n  selectSingleRow(row: T): void {\n    const rowId = this.getRowId(row);\n    const wasSelected = this.selectedRows.has(rowId);\n\n    this.selectedRows.clear();\n\n    if (!wasSelected) {\n      this.selectedRows.add(rowId);\n      this.emitSelectionChange('select', row);\n    } else {\n      this.emitSelectionChange('deselect', row);\n    }\n  }\n\n  private emitSelectionChange(\n    action: 'select' | 'deselect' | 'selectAll' | 'deselectAll',\n    changedRow?: T\n  ): void {\n    const selected = this.props.data.filter(row => this.isRowSelected(row));\n\n    this.selectionChange.emit({\n      selected,\n      changedRow,\n      action,\n    });\n  }\n\n  goToPage(page: number): void {\n    if (!this.props.pagination) return;\n\n    const previousPage = this.props.pagination.page;\n    const maxPage = this.totalPages - 1;\n    const newPage = Math.max(0, Math.min(page, maxPage));\n\n    this.pageChange.emit({\n      page: newPage,\n      pageSize: this.props.pagination.pageSize,\n      previousPage,\n    });\n\n    if (this.props.clientPagination) {\n      this.props.pagination.page = newPage;\n      this.updateDisplayedData();\n    }\n  }\n\n  onPageSizeChange(event: CustomEvent): void {\n    if (!this.props.pagination) return;\n\n    const newPageSize = event.detail.value;\n    const previousPage = this.props.pagination.page;\n\n    // Recalculate page to keep first visible item\n    const firstItem = previousPage * this.props.pagination.pageSize;\n    const newPage = Math.floor(firstItem / newPageSize);\n\n    this.pageChange.emit({\n      page: newPage,\n      pageSize: newPageSize,\n      previousPage,\n    });\n\n    if (this.props.clientPagination) {\n      this.props.pagination.pageSize = newPageSize;\n      this.props.pagination.page = newPage;\n      this.updateDisplayedData();\n    }\n  }\n}\n"]}
@@ -20887,9 +20887,16 @@ class DataTableComponent {
20887
20887
  this.emitSelectionChange(checked ? 'select' : 'deselect', row);
20888
20888
  }
20889
20889
  selectSingleRow(row) {
20890
+ const rowId = this.getRowId(row);
20891
+ const wasSelected = this.selectedRows.has(rowId);
20890
20892
  this.selectedRows.clear();
20891
- this.selectedRows.add(this.getRowId(row));
20892
- this.emitSelectionChange('select', row);
20893
+ if (!wasSelected) {
20894
+ this.selectedRows.add(rowId);
20895
+ this.emitSelectionChange('select', row);
20896
+ }
20897
+ else {
20898
+ this.emitSelectionChange('deselect', row);
20899
+ }
20893
20900
  }
20894
20901
  emitSelectionChange(action, changedRow) {
20895
20902
  const selected = this.props.data.filter(row => this.isRowSelected(row));
@@ -21085,8 +21092,7 @@ class DataTableComponent {
21085
21092
  <input
21086
21093
  type="radio"
21087
21094
  [checked]="isRowSelected(row)"
21088
- (change)="selectSingleRow(row)"
21089
- (click)="$event.stopPropagation()"
21095
+ (click)="selectSingleRow(row); $event.stopPropagation()"
21090
21096
  />
21091
21097
  }
21092
21098
  </td>
@@ -21164,8 +21170,7 @@ class DataTableComponent {
21164
21170
  <input
21165
21171
  type="radio"
21166
21172
  [checked]="isRowSelected(row)"
21167
- (change)="selectSingleRow(row)"
21168
- (click)="$event.stopPropagation()"
21173
+ (click)="selectSingleRow(row); $event.stopPropagation()"
21169
21174
  />
21170
21175
  }
21171
21176
  </div>
@@ -21439,8 +21444,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
21439
21444
  <input
21440
21445
  type="radio"
21441
21446
  [checked]="isRowSelected(row)"
21442
- (change)="selectSingleRow(row)"
21443
- (click)="$event.stopPropagation()"
21447
+ (click)="selectSingleRow(row); $event.stopPropagation()"
21444
21448
  />
21445
21449
  }
21446
21450
  </td>
@@ -21518,8 +21522,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
21518
21522
  <input
21519
21523
  type="radio"
21520
21524
  [checked]="isRowSelected(row)"
21521
- (change)="selectSingleRow(row)"
21522
- (click)="$event.stopPropagation()"
21525
+ (click)="selectSingleRow(row); $event.stopPropagation()"
21523
21526
  />
21524
21527
  }
21525
21528
  </div>