ng-prime-tools 1.0.33 → 1.0.34
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.mjs +5 -2
- package/fesm2022/ng-prime-tools.mjs +4 -1
- package/fesm2022/ng-prime-tools.mjs.map +1 -1
- package/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.d.ts.map +1 -1
- package/lib/pt-metric-panel/pt-metric-panel.component.d.ts +1 -1
- package/package.json +1 -1
|
@@ -142,7 +142,10 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
142
142
|
/** Called when a custom action button is clicked */
|
|
143
143
|
onCustomActionClick(action, row) {
|
|
144
144
|
if (action && typeof action.action === 'function') {
|
|
145
|
-
|
|
145
|
+
// Defer to next macrotask → fresh change detection cycle
|
|
146
|
+
setTimeout(() => {
|
|
147
|
+
action.action(row);
|
|
148
|
+
}, 0);
|
|
146
149
|
}
|
|
147
150
|
}
|
|
148
151
|
// Initialize filters for composed columns
|
|
@@ -502,4 +505,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
|
|
|
502
505
|
type: ViewChild,
|
|
503
506
|
args: ['dt', { static: false }]
|
|
504
507
|
}] } });
|
|
505
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-advanced-prime-table.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit, Output, ViewChild } from '@angular/core';\nimport { EventEmitter } from '@angular/core';\nimport { Table } from 'primeng/table';\nimport {\n  ImageStyle,\n  TableColumn,\n  TableTypeEnum,\n  TitleStyle,\n} from '../../public-api';\nimport { calculateTextWidth } from '../utils/text.util';\n\n/**\n * Generic action config passed from parent.\n *\n * Built-in codes:\n *  - 'edit'   → uses init/save callbacks and shows pencil / ✓ / ✗\n *  - 'delete' → uses callback(id) and shows trash icon\n *\n * Any other code is treated as a custom action, and its callback is invoked\n * directly with the whole row.\n */\nexport interface PTTableActionConfig {\n  code: string; // 'edit' | 'delete' | '<anything-you-want>'\n  icon?: string; // used for custom actions\n  styleClass?: string; // optional CSS classes for custom buttons\n  tooltip?: string; // optional tooltip (if you décide d'utiliser pTooltip)\n  action: any;\n}\n\n@Component({\n  selector: 'pt-advanced-prime-table',\n  templateUrl: './pt-advanced-prime-table.component.html',\n  styleUrl: './pt-advanced-prime-table.component.css',\n})\nexport class PTAdvancedPrimeTableComponent implements OnInit {\n  // Inputs\n  @Input() data: any[] = [];\n  @Input() columns: TableColumn[] = [];\n  @Input() totalRecords = 0;\n  @Input() rowsPerPage: number[] = [];\n  @Input() hasSearchFilter = false;\n  @Input() hasExportExcel = false;\n  @Input() hasExportPDF = false;\n  @Input() hasColumnFilter = false;\n  @Input() isPaginated = false;\n\n  /**\n   * Actions configuration.\n   * - code 'edit' / 'delete' garde l'ancien comportement.\n   * - tout autre code = action custom, gérée uniquement dans le parent.\n   */\n  @Input() actions: PTTableActionConfig[] = [];\n\n  @Input() isSortable = false;\n  @Input() loading = false;\n  @Input() maxHeight: string | null = null;\n\n  // Outputs: Events emitted to the parent component\n  @Output() filter = new EventEmitter<any>();\n  @Output() search = new EventEmitter<any>();\n  @Output() exportExcelEvent = new EventEmitter<void>();\n  @Output() exportPdfEvent = new EventEmitter<void>();\n  @Output() pageChange = new EventEmitter<{ page: number; rows: number }>();\n\n  @ViewChild('dt', { static: false }) dt!: Table;\n\n  public TableTypeEnum = TableTypeEnum;\n  searchValue = '';\n\n  public filters: { [key: string]: any } = {};\n\n  private validCurrencyCodes = ['USD', 'EUR', 'MAD'];\n  iconWidth = 77;\n\n  // Component state properties\n  rows = 0;\n  hasGroupedColumns = false;\n\n  // Flags / handlers for built-in edit/delete support\n  isDelete = false;\n  isEdit = false;\n\n  Delete: (value: any) => void = () => {};\n  initEditableRow: (data: any) => void = () => {};\n  saveEditableRow: (data: any) => void = () => {};\n  cancelEditableRow: (item: any) => void = () => {};\n\n  // Extra custom actions (codes other than 'edit'/'delete')\n  customActions: PTTableActionConfig[] = [];\n\n  // Data management properties\n  dataMap = new Map<any, any>();\n  map = new Map<any, Map<any, any>>();\n  optionEntries = new Map<string, any[]>();\n  optionValues: any[] = [];\n  globalFilterFields: string[] = [];\n\n  constructor() {}\n\n  ngOnInit(): void {\n    console.log('PTAdvancedPrimeTable columns:', this.columns);\n    console.log('PTAdvancedPrimeTable totalRecords input:', this.totalRecords);\n\n    this.hasGroupedColumns = this.columns.some(\n      (col) => col.children && col.children.length > 0\n    );\n\n    this.globalFilterFields = this.columns\n      .filter((col) => col.code !== undefined && col.isFilter !== false)\n      .map((col) => col.code as string);\n\n    this.initializePagination();\n    this.initializeActions();\n\n    // Set default value for isSortable / isEditable + widths\n    this.columns.forEach((col) => {\n      if (col.type === TableTypeEnum.ACTION) {\n        col.isEditable = false;\n        col.isFilter = false;\n        col.isSortable = false;\n      }\n\n      if (col.type === TableTypeEnum.COMPOSED) {\n        this.initializeComposedFilters(col);\n      }\n\n      if (col.isSortable === undefined) {\n        col.isSortable = true;\n      }\n\n      if (col.isEditable === undefined) {\n        col.isEditable = true;\n      }\n\n      if (col.isFilter !== false && col.code !== undefined) {\n        this.globalFilterFields.push(col.code);\n      }\n\n      if (!col.width) {\n        col.width = this.calculateColumnWidth(col);\n      }\n    });\n  }\n\n  // ---------- ACTIONS INITIALIZATION ----------\n\n  private initializeActions(): void {\n    // reset state\n    this.isDelete = false;\n    this.isEdit = false;\n    this.Delete = () => {};\n    this.initEditableRow = () => {};\n    this.saveEditableRow = () => {};\n    this.cancelEditableRow = () => {};\n    this.customActions = [];\n\n    if (!this.actions || this.actions.length === 0) {\n      return;\n    }\n\n    this.actions.forEach((action) => {\n      switch (action.code) {\n        case 'delete':\n          this.isDelete = true;\n          this.Delete = (value: any) => action.action(value);\n          break;\n\n        case 'edit':\n          this.initializeEditActions(action);\n          break;\n\n        default:\n          // any other code is a pure custom action\n          this.customActions.push(action);\n          break;\n      }\n    });\n  }\n\n  private initializeEditActions(action: PTTableActionConfig): void {\n    this.isEdit = true;\n    this.initEditableRow = (data: any) => action.action.init(data);\n    this.saveEditableRow = (data: any) => {\n      const record = this.map.get(data.id);\n      action.action.save(data, record);\n      this.dataMap.clear();\n    };\n    this.cancelEditableRow = (item: any) => console.log(item);\n  }\n\n  /** Called when a custom action button is clicked */\n  onCustomActionClick(action: PTTableActionConfig, row: any): void {\n    if (action && typeof action.action === 'function') {\n      action.action(row);\n    }\n  }\n\n  // Initialize filters for composed columns\n  private initializeComposedFilters(col: TableColumn): void {\n    col.composedNames?.forEach((composedName) => {\n      const code = (col.code as string) || '';\n      this.globalFilterFields.push(code + '.' + composedName);\n      this.filters[composedName] = {\n        options: col.filterOptions,\n        value: [],\n        label: 'Filter by ' + composedName,\n        placeholder: 'Select option',\n      };\n    });\n  }\n\n  // Get the column type for composed fields (STRING, IMAGE, etc.)\n  getComposedFieldType(\n    col: TableColumn,\n    composedName: string\n  ): TableTypeEnum | undefined {\n    if (col.composedNames && col.composedTypes) {\n      const index = col.composedNames.indexOf(composedName);\n      if (index >= 0 && index < col.composedTypes.length) {\n        return col.composedTypes[index];\n      }\n    }\n    return undefined;\n  }\n\n  onComposedFilterChange(\n    col: TableColumn,\n    composedName: string,\n    selectedValues: any\n  ): void {\n    this.filters[composedName].value = selectedValues;\n    this.filter.emit(this.filters);\n\n    const filterValue = selectedValues.join(',');\n    let matchResults: any[] = [];\n\n    const colCode = col.code as string;\n\n    selectedValues.forEach((value: any) => {\n      const matches = this.dt.value.filter((row: any) => {\n        return row[colCode]?.[composedName]\n          ?.toString()\n          .toLowerCase()\n          .includes(value.toLowerCase());\n      });\n      matchResults = [...matchResults, ...matches];\n    });\n\n    matchResults = Array.from(new Set(matchResults));\n    this.dt.filteredValue = matchResults;\n    this.dt.filterGlobal(filterValue, 'contains');\n  }\n\n  // ⛔ Don't override totalRecords here – we are in server-side mode\n  onFilter(event: any): void {\n    this.filter.emit(event);\n    // this.totalRecords = event.filteredValue?.length || 0;  // removed\n  }\n\n  onPageChange(event: any): void {\n    const page = event.page ?? Math.floor((event.first || 0) / event.rows);\n    const rows = event.rows;\n\n    this.rows = rows;\n    this.pageChange.emit({ page, rows });\n  }\n\n  onCalendarFilterChange(\n    event: any,\n    columnCode: string,\n    filterCallback: any\n  ): void {\n    const filterValue = event ? new Date(event) : null;\n    if (!filterValue) {\n      return;\n    }\n\n    filterCallback(filterValue);\n\n    const filterValueString = event ? this.formatDate(event) : '';\n    this.onFilter({\n      filteredValue: this.data.filter((item) => {\n        const columnValue = item[columnCode];\n        if (columnValue) {\n          const itemDateString = this.formatDate(new Date(columnValue));\n          return itemDateString === filterValueString;\n        }\n        return false;\n      }),\n    });\n  }\n\n  filterComposedData(item: any, composedName: string, value: any): boolean {\n    if (Array.isArray(value) && value.length > 0) {\n      return value.some((filterValue: string) =>\n        item[composedName]?.toLowerCase().includes(filterValue.toLowerCase())\n      );\n    }\n    return true;\n  }\n\n  private calculateColumnWidth(col: TableColumn): string {\n    const calculatedWidth = calculateTextWidth(col, col.title);\n    const totalWidth = calculatedWidth + this.iconWidth + 20;\n    return `${totalWidth}px`;\n  }\n\n  getHeaderWidth(col: TableColumn): string {\n    const base = col.width ?? this.calculateColumnWidth(col);\n    const widthWithoutPx = parseInt(base.replace('px', '') || '0', 10);\n    const headerWidth = widthWithoutPx + 20;\n    return `${headerWidth}px`;\n  }\n\n  clear(table: Table) {\n    table.clear();\n    this.searchValue = '';\n  }\n\n  private parseDate(dateString: string): Date | null {\n    const parts = dateString.split('/');\n    if (parts.length === 3) {\n      const day = parseInt(parts[0], 10);\n      const month = parseInt(parts[1], 10) - 1;\n      const year = parseInt(parts[2], 10);\n      const date = new Date(year, month, day);\n      return isNaN(date.getTime()) ? null : date;\n    }\n    return null;\n  }\n\n  private initializePagination(): void {\n    if (this.isPaginated) {\n      if (!this.rowsPerPage || this.rowsPerPage.length === 0) {\n        this.rowsPerPage = [20, 30, 40];\n      }\n      this.rows = this.rowsPerPage[0];\n    }\n  }\n\n  onChange(event: Event, id: number, key: any) {\n    const target = event.target as HTMLInputElement;\n    this.changeHandler(id, key, target.value);\n  }\n\n  changeHandler(id: number, key: any, value: any) {\n    const column = this.columns.find((item) => item.code === key);\n    if (!this.map.get(id)) {\n      if (column?.type === TableTypeEnum.DATE) {\n        const date = this.parseDate(value);\n        this.dataMap.set(key, date);\n      } else {\n        this.dataMap.set(key, value);\n      }\n      this.map.set(id, new Map(this.dataMap));\n    } else {\n      const mapItem = this.map.get(id) as Map<any, any>;\n      if (column?.type === TableTypeEnum.DATE) {\n        const date = this.parseDate(value);\n        mapItem.set(key, date);\n      } else {\n        mapItem.set(key, value);\n      }\n    }\n  }\n\n  getColumnFilterType(column: TableColumn): string {\n    switch (column.type) {\n      case TableTypeEnum.STRING:\n        return 'text';\n      case TableTypeEnum.AMOUNT:\n      case TableTypeEnum.NUMBER:\n        return 'numeric';\n      case TableTypeEnum.DATE:\n        return 'date';\n      case TableTypeEnum.MULTISELECT:\n        return 'multiSelect';\n      case TableTypeEnum.BOOLEAN:\n        return 'boolean';\n      case TableTypeEnum.COMPOSED:\n        return 'composed';\n      default:\n        return 'text';\n    }\n  }\n\n  isEditable(key: string | undefined): boolean {\n    if (!key) return false;\n    const column = this.columns.find((item) => item.code === key);\n    return column?.isEditable !== false;\n  }\n\n  isMultiSelect(key: any): boolean {\n    const column = this.columns.find((item) => item.code === key);\n    if (\n      column?.type === TableTypeEnum.MULTISELECT &&\n      column.options &&\n      column.code !== undefined\n    ) {\n      this.optionEntries = new Map([\n        [column.code as string, Object.values(column.options)],\n      ]);\n      this.optionValues = this.optionEntries.get(key) || [];\n      return true;\n    }\n    return false;\n  }\n\n  isDatePicker(key: any): boolean {\n    return (\n      this.columns.find((item) => item.code === key)?.type ===\n      TableTypeEnum.DATE\n    );\n  }\n\n  dateConverter(value: any): string {\n    return new Date(value).toLocaleDateString('en-US');\n  }\n\n  getCurrencySymbol(column: TableColumn): string | undefined {\n    return column.type === TableTypeEnum.AMOUNT &&\n      column.currency &&\n      this.isValidCurrencyCode(column.currency)\n      ? column.currency\n      : undefined;\n  }\n\n  private isValidCurrencyCode(currencyCode: string): boolean {\n    return this.validCurrencyCodes.includes(currencyCode);\n  }\n\n  filterGlobal(event: Event): void {\n    const target = event.target as HTMLInputElement;\n    const value = target.value.toLowerCase();\n\n    const filteredData = this.data.filter((item) => {\n      return this.globalFilterFields.some((field) => {\n        const column = this.columns.find((col) => col.code === field);\n        if (!column) {\n          return false;\n        }\n\n        if (column.type === TableTypeEnum.DATE) {\n          const itemDate = this.formatDate(item[field]);\n          return itemDate && itemDate.includes(value);\n        } else if (\n          column.type === TableTypeEnum.AMOUNT ||\n          column.type === TableTypeEnum.NUMBER\n        ) {\n          return (\n            item[field] && item[field].toString().toLowerCase().includes(value)\n          );\n        } else if (column.type === TableTypeEnum.COMPOSED) {\n          return this.filterComposedColumn(item[field], value);\n        } else {\n          return (\n            item[field] && item[field].toString().toLowerCase().includes(value)\n          );\n        }\n      });\n    });\n\n    this.dt.value = filteredData;\n\n    // ⛔ ne pas toucher à totalRecords en mode serveur\n    // this.totalRecords = filteredData.length ?? 0;\n  }\n\n  private filterComposedColumn(composedData: any, value: string): boolean {\n    if (composedData) {\n      return Object.keys(composedData).some((key) => {\n        const cellValue = composedData[key];\n        if (typeof cellValue === 'string') {\n          return cellValue.toLowerCase().includes(value);\n        }\n        return false;\n      });\n    }\n    return false;\n  }\n\n  formatDate(date: any): string {\n    if (!date) return '';\n    if (date instanceof Date) {\n      const day = date.getDate().toString().padStart(2, '0');\n      const month = (date.getMonth() + 1).toString().padStart(2, '0');\n      const year = date.getFullYear().toString();\n      return `${day}/${month}/${year}`;\n    } else if (typeof date === 'string') {\n      const parts = date.split('/');\n      if (parts.length === 3) {\n        return `${parts[0]}/${parts[1]}/${parts[2]}`;\n      } else {\n        return date;\n      }\n    }\n    return '';\n  }\n\n  exportExcel() {\n    this.exportExcelEvent.emit();\n  }\n\n  exportPdf() {\n    this.exportPdfEvent.emit();\n  }\n\n  getImageStyle(style: TitleStyle | ImageStyle | undefined): {\n    [key: string]: string;\n  } {\n    if (style) {\n      const imageStyle: { [key: string]: string } = {\n        width: (style as ImageStyle).width || 'auto',\n        height: (style as ImageStyle).height || 'auto',\n      };\n\n      if ((style as ImageStyle).margin) {\n        imageStyle.margin = (style as ImageStyle).margin!;\n      }\n      if ((style as ImageStyle).marginLeft) {\n        imageStyle.marginLeft = (style as ImageStyle).marginLeft!;\n      }\n      if ((style as ImageStyle).marginRight) {\n        imageStyle.marginRight = (style as ImageStyle).marginRight!;\n      }\n      if ((style as ImageStyle).marginTop) {\n        imageStyle.marginTop = (style as ImageStyle).marginTop!;\n      }\n      if ((style as ImageStyle).marginBottom) {\n        imageStyle.marginBottom = (style as ImageStyle).marginBottom!;\n      }\n\n      return imageStyle;\n    }\n\n    return {};\n  }\n\n  getTitleStyle(style: TitleStyle | ImageStyle | undefined): {\n    [key: string]: string;\n  } {\n    if (style) {\n      return {\n        color: style.color || 'inherit',\n        fontSize: (style as TitleStyle).fontSize || 'inherit',\n        textAlign: (style as TitleStyle).position || 'left',\n      };\n    }\n    return {};\n  }\n\n  formatNumber(\n    value: number,\n    decimalPlaces?: number,\n    thousandSeparator: 'comma' | 'space' = 'comma',\n    decimalSeparator: 'comma' | 'dot' = 'dot'\n  ): string {\n    if (value === null || value === undefined || isNaN(value)) return '';\n\n    let formattedNumber =\n      decimalPlaces !== undefined\n        ? value.toFixed(decimalPlaces)\n        : value.toString();\n\n    if (decimalSeparator === 'comma') {\n      formattedNumber = formattedNumber.replace('.', ',');\n    }\n\n    if (thousandSeparator && Math.abs(value) >= 1000) {\n      const parts = formattedNumber.split(\n        decimalSeparator === 'comma' ? ',' : '.'\n      );\n      parts[0] = parts[0].replace(\n        /\\B(?=(\\d{3})+(?!\\d))/g,\n        thousandSeparator === 'comma' ? ',' : ' '\n      );\n      formattedNumber = parts.join(decimalSeparator === 'comma' ? ',' : '.');\n    }\n\n    return formattedNumber;\n  }\n}\n","<div class=\"pt-advanced-prime-table table-container\">\n  <p-table\n    #dt\n    [value]=\"data\"\n    [loading]=\"loading\"\n    [rows]=\"rows\"\n    [paginator]=\"isPaginated\"\n    [globalFilterFields]=\"globalFilterFields\"\n    [rowsPerPageOptions]=\"rowsPerPage\"\n    [totalRecords]=\"totalRecords\"\n    [lazy]=\"true\"\n    dataKey=\"id\"\n    styleClass=\"p-datatable-gridlines p-datatable-striped\"\n    [scrollable]=\"true\"\n    [scrollHeight]=\"maxHeight !== null ? maxHeight : undefined\"\n    (onFilter)=\"onFilter($event)\"\n    (onPage)=\"onPageChange($event)\"\n  >\n    <!-- CAPTION -->\n    <ng-template pTemplate=\"caption\">\n      <div class=\"flex\">\n        <div>\n          <h3>Total: {{ totalRecords }}</h3>\n        </div>\n\n        <div>\n          <!-- Clear filters -->\n          <button\n            *ngIf=\"hasSearchFilter\"\n            pButton\n            icon=\"pi pi-filter-slash\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"clear(dt)\"\n            title=\"Clear filters\"\n          ></button>\n\n          <!-- Export to Excel Button -->\n          <button\n            *ngIf=\"hasExportExcel\"\n            pButton\n            icon=\"pi pi-file-excel\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"exportExcel()\"\n            title=\"Export to Excel\"\n          ></button>\n\n          <!-- Export to PDF Button -->\n          <button\n            *ngIf=\"hasExportPDF\"\n            pButton\n            icon=\"pi pi-file-pdf\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"exportPdf()\"\n            title=\"Export to PDF\"\n          ></button>\n        </div>\n\n        <div class=\"ml-auto\" *ngIf=\"hasSearchFilter\">\n          <p-iconField iconPosition=\"left\" class=\"ml-auto\">\n            <p-inputIcon>\n              <i class=\"pi pi-search\"></i>\n            </p-inputIcon>\n            <input\n              pInputText\n              type=\"text\"\n              [(ngModel)]=\"searchValue\"\n              (input)=\"filterGlobal($event)\"\n              placeholder=\"Search keyword\"\n            />\n          </p-iconField>\n        </div>\n      </div>\n    </ng-template>\n\n    <!-- HEADER -->\n    <ng-template pTemplate=\"header\">\n      <tr class=\"sticky-header\">\n        <ng-container *ngFor=\"let col of columns\">\n          <th\n            *ngIf=\"!col.children; else groupHeader\"\n            [style.width]=\"getHeaderWidth(col)\"\n            [style.padding]=\"'0px'\"\n            colspan=\"1\"\n          >\n            <ng-container\n              *ngIf=\"isSortable && col.isSortable !== false; else noSortHeader\"\n            >\n              <th\n                pSortableColumn=\"{{ col.code }}\"\n                [style.width]=\"getHeaderWidth(col)\"\n              >\n                <div\n                  class=\"header-container d-flex align-items-center justify-content-between\"\n                  [style.width]=\"col.width\"\n                  [style.padding]=\"'0px'\"\n                  [style.margin]=\"'10px'\"\n                >\n                  <span>{{ col.title }}</span>\n                  <div\n                    class=\"icons d-flex align-items-center\"\n                    [style.width]=\"'77px'\"\n                  >\n                    <p-sortIcon field=\"{{ col.code }}\" />\n                    <ng-container *ngIf=\"col.isFilter !== false\">\n                      <!-- COMPOSED FILTER -->\n                      <p-columnFilter\n                        display=\"menu\"\n                        [field]=\"col.code\"\n                        [type]=\"getColumnFilterType(col)\"\n                        *ngIf=\"col.type === TableTypeEnum.COMPOSED\"\n                        showClearButton=\"false\"\n                        showApplyButton=\"false\"\n                      >\n                        <ng-template\n                          pTemplate=\"filter\"\n                          let-value\n                          let-filterCallback=\"filterCallback\"\n                          (onFilter)=\"onFilter($event)\"\n                        >\n                          <div *ngFor=\"let composedName of col.composedNames\">\n                            <ng-container\n                              *ngIf=\"\n                                getComposedFieldType(col, composedName) ===\n                                TableTypeEnum.STRING\n                              \"\n                            >\n                              <p-multiSelect\n                                [ngModel]=\"filters[composedName]?.value\"\n                                [options]=\"filters[composedName]?.options\"\n                                (onChange)=\"\n                                  onComposedFilterChange(\n                                    col,\n                                    composedName,\n                                    $event.value\n                                  )\n                                \"\n                                [placeholder]=\"\n                                  filters[composedName]?.placeholder\n                                \"\n                                [display]=\"'chip'\"\n                              >\n                                <ng-template let-item pTemplate=\"item\">\n                                  <div class=\"custom-multiselect-item\">\n                                    <img\n                                      *ngIf=\"item.image\"\n                                      [src]=\"item.image\"\n                                      alt=\"icon\"\n                                      class=\"filter-image\"\n                                    />\n                                    <span>{{ item.label }}</span>\n                                  </div>\n                                </ng-template>\n                              </p-multiSelect>\n                            </ng-container>\n                          </div>\n\n                          <ng-template let-item pTemplate=\"item\">\n                            <div class=\"custom-multiselect-item\">\n                              <img\n                                *ngIf=\"item.image\"\n                                [src]=\"item.image\"\n                                alt=\"icon\"\n                                class=\"filter-image\"\n                              />\n                              <span>{{ item.label }}</span>\n                            </div>\n                          </ng-template>\n                        </ng-template>\n                      </p-columnFilter>\n\n                      <!-- OTHER TYPES -->\n                      <p-columnFilter\n                        display=\"menu\"\n                        [field]=\"col.code\"\n                        [type]=\"getColumnFilterType(col)\"\n                        *ngIf=\"col.type !== TableTypeEnum.COMPOSED\"\n                        hideOnClear=\"true\"\n                      >\n                        <!-- NUMBER -->\n                        <ng-template\n                          pTemplate=\"filter\"\n                          let-value\n                          let-filterCallback=\"filterCallback\"\n                          *ngIf=\"col.type === TableTypeEnum.NUMBER\"\n                          (onFilter)=\"onFilter($event)\"\n                        >\n                          <input\n                            pInputText\n                            type=\"number\"\n                            [step]=\"\n                              col.decimalPlaces\n                                ? '0.' + '1'.padEnd(col.decimalPlaces, '0')\n                                : 'any'\n                            \"\n                            [ngModel]=\"value\"\n                            (ngModelChange)=\"filterCallback($event)\"\n                            placeholder=\"Enter a number\"\n                          />\n                        </ng-template>\n\n                        <!-- DATE -->\n                        <ng-template\n                          pTemplate=\"filter\"\n                          let-value\n                          let-filterCallback=\"filterCallback\"\n                          *ngIf=\"col.type === TableTypeEnum.DATE\"\n                          (onFilter)=\"onFilter($event)\"\n                        >\n                          <p-calendar\n                            [ngModel]=\"value\"\n                            (ngModelChange)=\"\n                              onCalendarFilterChange(\n                                $event,\n                                col.code!,\n                                filterCallback\n                              )\n                            \"\n                            dateFormat=\"dd/mm/yy\"\n                            placeholder=\"Choose a date\"\n                          ></p-calendar>\n                        </ng-template>\n\n                        <!-- MULTISELECT -->\n                        <ng-template\n                          pTemplate=\"filter\"\n                          let-value\n                          let-filterCallback=\"filterCallback\"\n                          *ngIf=\"\n                            col.type === TableTypeEnum.MULTISELECT &&\n                            col.filterOptions &&\n                            col.filterOptions.length > 0\n                          \"\n                          (onFilter)=\"onFilter($event)\"\n                        >\n                          <p-multiSelect\n                            [options]=\"col.filterOptions\"\n                            [ngModel]=\"value\"\n                            (ngModelChange)=\"filterCallback($event)\"\n                            display=\"chip\"\n                            placeholder=\"Choose option\"\n                            class=\"custom-multiselect\"\n                          ></p-multiSelect>\n                        </ng-template>\n                      </p-columnFilter>\n                    </ng-container>\n                  </div>\n                </div>\n              </th>\n            </ng-container>\n\n            <ng-template #noSortHeader>\n              <th>\n                <div class=\"header-container\">\n                  <span>{{ col.title }}</span>\n                  <ng-container *ngIf=\"col.isFilter !== false\">\n                    <p-columnFilter\n                      *ngIf=\"col.type === 'AMOUNT'\"\n                      display=\"menu\"\n                      [field]=\"col.code\"\n                      [type]=\"getColumnFilterType(col)\"\n                      [currency]=\"getCurrencySymbol(col)\"\n                    ></p-columnFilter>\n\n                    <p-columnFilter\n                      *ngIf=\"col.type !== 'AMOUNT'\"\n                      display=\"menu\"\n                      [field]=\"col.code\"\n                      [type]=\"getColumnFilterType(col)\"\n                    >\n                      <ng-template\n                        pTemplate=\"filter\"\n                        let-value\n                        let-filterCallback=\"filterCallback\"\n                        *ngIf=\"getColumnFilterType(col) === 'date'\"\n                      >\n                        <p-calendar\n                          [ngModel]=\"value\"\n                          (ngModelChange)=\"filterCallback($event)\"\n                          dateFormat=\"dd/mm/yy\"\n                        ></p-calendar>\n                      </ng-template>\n\n                      <ng-template\n                        pTemplate=\"filter\"\n                        let-value\n                        let-filterCallback=\"filterCallback\"\n                        *ngIf=\"getColumnFilterType(col) === 'multiSelect'\"\n                      >\n                        <p-multiSelect\n                          [options]=\"col.filterOptions\"\n                          [ngModel]=\"value\"\n                          (ngModelChange)=\"filterCallback($event)\"\n                          display=\"chip\"\n                          placeholder=\"Select\"\n                          class=\"custom-multiselect\"\n                        ></p-multiSelect>\n                      </ng-template>\n                    </p-columnFilter>\n                  </ng-container>\n                </div>\n              </th>\n            </ng-template>\n          </th>\n\n          <!-- GROUPED HEADER -->\n          <ng-template #groupHeader>\n            <th\n              [attr.colspan]=\"col.children?.length\"\n              [style.width]=\"getHeaderWidth(col)\"\n              [style.text-align]=\"'center'\"\n            >\n              <span>{{ col.title }}</span>\n            </th>\n          </ng-template>\n        </ng-container>\n      </tr>\n\n      <!-- CHILD HEADERS -->\n      <tr *ngIf=\"hasGroupedColumns\">\n        <ng-container *ngFor=\"let col of columns\">\n          <ng-container *ngIf=\"col.children\">\n            <th\n              *ngFor=\"let child of col.children\"\n              [style.width]=\"getHeaderWidth(child)\"\n              [style.padding]=\"'0px'\"\n            ></th>\n          </ng-container>\n        </ng-container>\n      </tr>\n    </ng-template>\n\n    <!-- EMPTY MESSAGE -->\n    <ng-template pTemplate=\"emptymessage\">\n      <div class=\"empty-message\">\n        <i class=\"pi pi-info-circle\"></i>\n        <p>No records available to display.</p>\n      </div>\n    </ng-template>\n\n    <!-- BODY -->\n    <ng-template\n      pTemplate=\"body\"\n      let-data\n      let-editing=\"editing\"\n      let-ri=\"rowIndex\"\n    >\n      <tr *ngIf=\"!loading\" [pEditableRow]=\"isEdit ? data : null\">\n        <ng-container *ngFor=\"let col of columns\">\n          <ng-container *ngIf=\"!col.children; else childColumns\">\n            <!-- show cell when:\n                 - ACTION column (no value needed)\n                 - OR normal col with value present -->\n            <ng-container\n              *ngIf=\"\n                col.type === TableTypeEnum.ACTION ||\n                (col.code !== undefined && data[col.code!] !== undefined)\n              \"\n            >\n              <td\n                *ngIf=\"\n                  isEditable(col.code!) && col.type !== TableTypeEnum.ACTION;\n                  else normalTD\n                \"\n                [style.width]=\"getHeaderWidth(col)\"\n              >\n                <!-- Editable input for NUMBER/DATE/STRING/MULTISELECT -->\n                <ng-container *ngIf=\"isMultiSelect(col.code); else datePicker\">\n                  <p-cellEditor>\n                    <ng-template pTemplate=\"input\">\n                      <p-multiSelect\n                        appendTo=\"body\"\n                        [ngModel]=\"data[col.code!]\"\n                        [style]=\"{ width: '100%' }\"\n                        (ngModelChange)=\"\n                          changeHandler(data.id, col.code, $event)\n                        \"\n                        [options]=\"optionValues\"\n                      ></p-multiSelect>\n                    </ng-template>\n                    <ng-template pTemplate=\"output\">\n                      <div class=\"multi-select-container\">\n                        <ng-container *ngFor=\"let rec of data[col.code!]\">\n                          <p-tag [value]=\"rec\"></p-tag>\n                        </ng-container>\n                      </div>\n                    </ng-template>\n                  </p-cellEditor>\n                </ng-container>\n\n                <ng-template #datePicker>\n                  <ng-container\n                    *ngIf=\"isDatePicker(col.code); else normalInput\"\n                  >\n                    <p-cellEditor>\n                      <ng-template pTemplate=\"input\">\n                        <p-calendar\n                          [inputId]=\"data[col.code!]\"\n                          [ngModel]=\"data[col.code!]\"\n                          (ngModelChange)=\"\n                            changeHandler(data.id, col.code, $event)\n                          \"\n                          [dateFormat]=\"'dd/mm/yy'\"\n                        ></p-calendar>\n                      </ng-template>\n                      <ng-template pTemplate=\"output\">\n                        {{ data[col.code!] | customDate }}\n                      </ng-template>\n                    </p-cellEditor>\n                  </ng-container>\n                </ng-template>\n\n                <ng-template #normalInput>\n                  <p-cellEditor>\n                    <ng-template pTemplate=\"input\">\n                      <input\n                        pInputText\n                        type=\"text\"\n                        [ngModel]=\"data[col.code!]\"\n                        (change)=\"onChange($event, data.id, col.code)\"\n                      />\n                    </ng-template>\n                    <ng-template pTemplate=\"output\">\n                      <ng-container\n                        *ngIf=\"\n                          col.type === TableTypeEnum.AMOUNT;\n                          else normalOutput\n                        \"\n                      >\n                        {{\n                          data[col.code!]\n                            | customCurrency\n                              : getCurrencySymbol(col)\n                              : col.decimalPlaces\n                              : col.thousandSeparator\n                              : col.decimalSeparator\n                        }}\n                      </ng-container>\n                      <ng-template #normalOutput>\n                        {{ data[col.code!] }}\n                      </ng-template>\n                    </ng-template>\n                  </p-cellEditor>\n                </ng-template>\n              </td>\n\n              <ng-template #normalTD>\n                <td [style.width]=\"getHeaderWidth(col)\">\n                  <!-- ACTION column: built-in edit/delete + custom actions -->\n                  <ng-container\n                    *ngIf=\"\n                      col.type === TableTypeEnum.ACTION;\n                      else nonActionCell\n                    \"\n                  >\n                    <div class=\"action-buttons-container\">\n                      <!-- built-in delete -->\n                      <button\n                        *ngIf=\"isDelete\"\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        icon=\"pi pi-trash\"\n                        (click)=\"Delete(data.id)\"\n                        class=\"p-button-rounded p-button-text\"\n                      ></button>\n\n                      <!-- built-in inline edit (old behaviour) -->\n                      <div *ngIf=\"isEdit\">\n                        <button\n                          pInitEditableRow\n                          *ngIf=\"!editing\"\n                          pButton\n                          pRipple\n                          type=\"button\"\n                          icon=\"pi pi-pencil\"\n                          (click)=\"initEditableRow(data)\"\n                          class=\"p-button-rounded p-button-text\"\n                        ></button>\n                        <button\n                          *ngIf=\"editing\"\n                          pSaveEditableRow\n                          pButton\n                          pRipple\n                          type=\"button\"\n                          icon=\"pi pi-check\"\n                          (click)=\"saveEditableRow(data)\"\n                          class=\"p-button-rounded p-button-text\"\n                        ></button>\n                        <button\n                          *ngIf=\"editing\"\n                          pCancelEditableRow\n                          pButton\n                          pRipple\n                          type=\"button\"\n                          icon=\"pi pi-times\"\n                          (click)=\"cancelEditableRow(data)\"\n                          class=\"p-button-rounded p-button-text\"\n                        ></button>\n                      </div>\n\n                      <!-- custom actions -->\n                      <button\n                        *ngFor=\"let act of customActions\"\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        class=\"p-button-rounded p-button-text\"\n                        [icon]=\"act.icon || 'pi pi-ellipsis-h'\"\n                        [ngClass]=\"act.styleClass\"\n                        (click)=\"onCustomActionClick(act, data)\"\n                      ></button>\n                    </div>\n                  </ng-container>\n\n                  <!-- NON-ACTION cells -->\n                  <ng-template #nonActionCell>\n                    <!-- COMPOSED -->\n                    <ng-container\n                      *ngIf=\"\n                        col.type === TableTypeEnum.COMPOSED;\n                        else nonComposed\n                      \"\n                    >\n                      <div class=\"composed-cell\">\n                        <ng-container\n                          *ngFor=\"\n                            let composedName of col.composedNames;\n                            let i = index\n                          \"\n                        >\n                          <!-- IMAGE -->\n                          <ng-container\n                            *ngIf=\"\n                              col.composedTypes &&\n                              col.composedTypes[i] === TableTypeEnum.IMAGE\n                            \"\n                          >\n                            <img\n                              [src]=\"data[col.code!][composedName]\"\n                              alt=\"composed-img\"\n                              class=\"composed-image\"\n                              [ngStyle]=\"\n                                getImageStyle(\n                                  col.composedStyles?.[composedName]\n                                )\n                              \"\n                            />\n                          </ng-container>\n\n                          <!-- STRING -->\n                          <ng-container\n                            *ngIf=\"\n                              col.composedTypes &&\n                              col.composedTypes[i] === TableTypeEnum.STRING\n                            \"\n                          >\n                            <span\n                              class=\"composed-text\"\n                              [ngStyle]=\"\n                                getTitleStyle(\n                                  col.composedStyles?.[composedName]\n                                )\n                              \"\n                            >\n                              {{ data[col.code!][composedName] }}\n                            </span>\n                          </ng-container>\n                        </ng-container>\n                      </div>\n                    </ng-container>\n\n                    <ng-template #nonComposed>\n                      <!-- AMOUNT-->\n                      <ng-container\n                        *ngIf=\"\n                          col.type === TableTypeEnum.AMOUNT;\n                          else nonAmount\n                        \"\n                      >\n                        {{\n                          data[col.code!]\n                            | customCurrency\n                              : getCurrencySymbol(col)\n                              : col.decimalPlaces\n                              : col.thousandSeparator\n                              : col.decimalSeparator\n                        }}\n                      </ng-container>\n\n                      <ng-template #nonAmount>\n                        <!-- NUMBER-->\n                        <ng-container\n                          *ngIf=\"\n                            col.type === TableTypeEnum.NUMBER;\n                            else nonNumber\n                          \"\n                        >\n                          {{\n                            formatNumber(\n                              data[col.code!],\n                              col.decimalPlaces,\n                              col.thousandSeparator,\n                              col.decimalSeparator\n                            )\n                          }}\n                        </ng-container>\n\n                        <ng-template #nonNumber>\n                          <!-- DATE -->\n                          <ng-container\n                            *ngIf=\"\n                              col.type === TableTypeEnum.DATE;\n                              else normalTypes\n                            \"\n                          >\n                            {{ formatDate(data[col.code!]) }}\n                          </ng-container>\n\n                          <ng-template #normalTypes>\n                            <!-- STRING, MULTISELECT-->\n                            <ng-container\n                              *ngIf=\"\n                                [\n                                  TableTypeEnum.STRING,\n                                  TableTypeEnum.MULTISELECT\n                                ].includes(col.type!)\n                              \"\n                            >\n                              {{ data[col.code!] }}\n                            </ng-container>\n                          </ng-template>\n                        </ng-template>\n                      </ng-template>\n                    </ng-template>\n                  </ng-template>\n                </td>\n              </ng-template>\n            </ng-container>\n          </ng-container>\n\n          <!-- CHILD COLUMNS -->\n          <ng-template #childColumns>\n            <ng-container *ngFor=\"let child of col.children\">\n              <ng-container\n                *ngIf=\"\n                  child.code !== undefined && data[child.code] !== undefined\n                \"\n              >\n                <td [style.width]=\"getHeaderWidth(child)\">\n                  <ng-container\n                    *ngIf=\"isEditable(child.code); else childNormalTD\"\n                  >\n                    <p-cellEditor>\n                      <ng-template pTemplate=\"input\">\n                        <input\n                          pInputText\n                          type=\"text\"\n                          [ngModel]=\"data[child.code]\"\n                          (change)=\"onChange($event, data.id, child.code)\"\n                        />\n                      </ng-template>\n                      <ng-template pTemplate=\"output\">\n                        {{ data[child.code] }}\n                      </ng-template>\n                    </p-cellEditor>\n                  </ng-container>\n\n                  <ng-template #childNormalTD>\n                    {{ data[child.code] }}\n                  </ng-template>\n                </td>\n              </ng-container>\n            </ng-container>\n          </ng-template>\n        </ng-container>\n      </tr>\n    </ng-template>\n  </p-table>\n</div>\n"]}
|
|
508
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-advanced-prime-table.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit, Output, ViewChild } from '@angular/core';\nimport { EventEmitter } from '@angular/core';\nimport { Table } from 'primeng/table';\nimport {\n  ImageStyle,\n  TableColumn,\n  TableTypeEnum,\n  TitleStyle,\n} from '../../public-api';\nimport { calculateTextWidth } from '../utils/text.util';\n\n/**\n * Generic action config passed from parent.\n *\n * Built-in codes:\n *  - 'edit'   → uses init/save callbacks and shows pencil / ✓ / ✗\n *  - 'delete' → uses callback(id) and shows trash icon\n *\n * Any other code is treated as a custom action, and its callback is invoked\n * directly with the whole row.\n */\nexport interface PTTableActionConfig {\n  code: string; // 'edit' | 'delete' | '<anything-you-want>'\n  icon?: string; // used for custom actions\n  styleClass?: string; // optional CSS classes for custom buttons\n  tooltip?: string; // optional tooltip (if you décide d'utiliser pTooltip)\n  action: any;\n}\n\n@Component({\n  selector: 'pt-advanced-prime-table',\n  templateUrl: './pt-advanced-prime-table.component.html',\n  styleUrl: './pt-advanced-prime-table.component.css',\n})\nexport class PTAdvancedPrimeTableComponent implements OnInit {\n  // Inputs\n  @Input() data: any[] = [];\n  @Input() columns: TableColumn[] = [];\n  @Input() totalRecords = 0;\n  @Input() rowsPerPage: number[] = [];\n  @Input() hasSearchFilter = false;\n  @Input() hasExportExcel = false;\n  @Input() hasExportPDF = false;\n  @Input() hasColumnFilter = false;\n  @Input() isPaginated = false;\n\n  /**\n   * Actions configuration.\n   * - code 'edit' / 'delete' garde l'ancien comportement.\n   * - tout autre code = action custom, gérée uniquement dans le parent.\n   */\n  @Input() actions: PTTableActionConfig[] = [];\n\n  @Input() isSortable = false;\n  @Input() loading = false;\n  @Input() maxHeight: string | null = null;\n\n  // Outputs: Events emitted to the parent component\n  @Output() filter = new EventEmitter<any>();\n  @Output() search = new EventEmitter<any>();\n  @Output() exportExcelEvent = new EventEmitter<void>();\n  @Output() exportPdfEvent = new EventEmitter<void>();\n  @Output() pageChange = new EventEmitter<{ page: number; rows: number }>();\n\n  @ViewChild('dt', { static: false }) dt!: Table;\n\n  public TableTypeEnum = TableTypeEnum;\n  searchValue = '';\n\n  public filters: { [key: string]: any } = {};\n\n  private validCurrencyCodes = ['USD', 'EUR', 'MAD'];\n  iconWidth = 77;\n\n  // Component state properties\n  rows = 0;\n  hasGroupedColumns = false;\n\n  // Flags / handlers for built-in edit/delete support\n  isDelete = false;\n  isEdit = false;\n\n  Delete: (value: any) => void = () => {};\n  initEditableRow: (data: any) => void = () => {};\n  saveEditableRow: (data: any) => void = () => {};\n  cancelEditableRow: (item: any) => void = () => {};\n\n  // Extra custom actions (codes other than 'edit'/'delete')\n  customActions: PTTableActionConfig[] = [];\n\n  // Data management properties\n  dataMap = new Map<any, any>();\n  map = new Map<any, Map<any, any>>();\n  optionEntries = new Map<string, any[]>();\n  optionValues: any[] = [];\n  globalFilterFields: string[] = [];\n\n  constructor() {}\n\n  ngOnInit(): void {\n    console.log('PTAdvancedPrimeTable columns:', this.columns);\n    console.log('PTAdvancedPrimeTable totalRecords input:', this.totalRecords);\n\n    this.hasGroupedColumns = this.columns.some(\n      (col) => col.children && col.children.length > 0\n    );\n\n    this.globalFilterFields = this.columns\n      .filter((col) => col.code !== undefined && col.isFilter !== false)\n      .map((col) => col.code as string);\n\n    this.initializePagination();\n    this.initializeActions();\n\n    // Set default value for isSortable / isEditable + widths\n    this.columns.forEach((col) => {\n      if (col.type === TableTypeEnum.ACTION) {\n        col.isEditable = false;\n        col.isFilter = false;\n        col.isSortable = false;\n      }\n\n      if (col.type === TableTypeEnum.COMPOSED) {\n        this.initializeComposedFilters(col);\n      }\n\n      if (col.isSortable === undefined) {\n        col.isSortable = true;\n      }\n\n      if (col.isEditable === undefined) {\n        col.isEditable = true;\n      }\n\n      if (col.isFilter !== false && col.code !== undefined) {\n        this.globalFilterFields.push(col.code);\n      }\n\n      if (!col.width) {\n        col.width = this.calculateColumnWidth(col);\n      }\n    });\n  }\n\n  // ---------- ACTIONS INITIALIZATION ----------\n\n  private initializeActions(): void {\n    // reset state\n    this.isDelete = false;\n    this.isEdit = false;\n    this.Delete = () => {};\n    this.initEditableRow = () => {};\n    this.saveEditableRow = () => {};\n    this.cancelEditableRow = () => {};\n    this.customActions = [];\n\n    if (!this.actions || this.actions.length === 0) {\n      return;\n    }\n\n    this.actions.forEach((action) => {\n      switch (action.code) {\n        case 'delete':\n          this.isDelete = true;\n          this.Delete = (value: any) => action.action(value);\n          break;\n\n        case 'edit':\n          this.initializeEditActions(action);\n          break;\n\n        default:\n          // any other code is a pure custom action\n          this.customActions.push(action);\n          break;\n      }\n    });\n  }\n\n  private initializeEditActions(action: PTTableActionConfig): void {\n    this.isEdit = true;\n    this.initEditableRow = (data: any) => action.action.init(data);\n    this.saveEditableRow = (data: any) => {\n      const record = this.map.get(data.id);\n      action.action.save(data, record);\n      this.dataMap.clear();\n    };\n    this.cancelEditableRow = (item: any) => console.log(item);\n  }\n\n  /** Called when a custom action button is clicked */\n  onCustomActionClick(action: PTTableActionConfig, row: any): void {\n    if (action && typeof action.action === 'function') {\n      // Defer to next macrotask → fresh change detection cycle\n      setTimeout(() => {\n        action.action(row);\n      }, 0);\n    }\n  }\n\n  // Initialize filters for composed columns\n  private initializeComposedFilters(col: TableColumn): void {\n    col.composedNames?.forEach((composedName) => {\n      const code = (col.code as string) || '';\n      this.globalFilterFields.push(code + '.' + composedName);\n      this.filters[composedName] = {\n        options: col.filterOptions,\n        value: [],\n        label: 'Filter by ' + composedName,\n        placeholder: 'Select option',\n      };\n    });\n  }\n\n  // Get the column type for composed fields (STRING, IMAGE, etc.)\n  getComposedFieldType(\n    col: TableColumn,\n    composedName: string\n  ): TableTypeEnum | undefined {\n    if (col.composedNames && col.composedTypes) {\n      const index = col.composedNames.indexOf(composedName);\n      if (index >= 0 && index < col.composedTypes.length) {\n        return col.composedTypes[index];\n      }\n    }\n    return undefined;\n  }\n\n  onComposedFilterChange(\n    col: TableColumn,\n    composedName: string,\n    selectedValues: any\n  ): void {\n    this.filters[composedName].value = selectedValues;\n    this.filter.emit(this.filters);\n\n    const filterValue = selectedValues.join(',');\n    let matchResults: any[] = [];\n\n    const colCode = col.code as string;\n\n    selectedValues.forEach((value: any) => {\n      const matches = this.dt.value.filter((row: any) => {\n        return row[colCode]?.[composedName]\n          ?.toString()\n          .toLowerCase()\n          .includes(value.toLowerCase());\n      });\n      matchResults = [...matchResults, ...matches];\n    });\n\n    matchResults = Array.from(new Set(matchResults));\n    this.dt.filteredValue = matchResults;\n    this.dt.filterGlobal(filterValue, 'contains');\n  }\n\n  // ⛔ Don't override totalRecords here – we are in server-side mode\n  onFilter(event: any): void {\n    this.filter.emit(event);\n    // this.totalRecords = event.filteredValue?.length || 0;  // removed\n  }\n\n  onPageChange(event: any): void {\n    const page = event.page ?? Math.floor((event.first || 0) / event.rows);\n    const rows = event.rows;\n\n    this.rows = rows;\n    this.pageChange.emit({ page, rows });\n  }\n\n  onCalendarFilterChange(\n    event: any,\n    columnCode: string,\n    filterCallback: any\n  ): void {\n    const filterValue = event ? new Date(event) : null;\n    if (!filterValue) {\n      return;\n    }\n\n    filterCallback(filterValue);\n\n    const filterValueString = event ? this.formatDate(event) : '';\n    this.onFilter({\n      filteredValue: this.data.filter((item) => {\n        const columnValue = item[columnCode];\n        if (columnValue) {\n          const itemDateString = this.formatDate(new Date(columnValue));\n          return itemDateString === filterValueString;\n        }\n        return false;\n      }),\n    });\n  }\n\n  filterComposedData(item: any, composedName: string, value: any): boolean {\n    if (Array.isArray(value) && value.length > 0) {\n      return value.some((filterValue: string) =>\n        item[composedName]?.toLowerCase().includes(filterValue.toLowerCase())\n      );\n    }\n    return true;\n  }\n\n  private calculateColumnWidth(col: TableColumn): string {\n    const calculatedWidth = calculateTextWidth(col, col.title);\n    const totalWidth = calculatedWidth + this.iconWidth + 20;\n    return `${totalWidth}px`;\n  }\n\n  getHeaderWidth(col: TableColumn): string {\n    const base = col.width ?? this.calculateColumnWidth(col);\n    const widthWithoutPx = parseInt(base.replace('px', '') || '0', 10);\n    const headerWidth = widthWithoutPx + 20;\n    return `${headerWidth}px`;\n  }\n\n  clear(table: Table) {\n    table.clear();\n    this.searchValue = '';\n  }\n\n  private parseDate(dateString: string): Date | null {\n    const parts = dateString.split('/');\n    if (parts.length === 3) {\n      const day = parseInt(parts[0], 10);\n      const month = parseInt(parts[1], 10) - 1;\n      const year = parseInt(parts[2], 10);\n      const date = new Date(year, month, day);\n      return isNaN(date.getTime()) ? null : date;\n    }\n    return null;\n  }\n\n  private initializePagination(): void {\n    if (this.isPaginated) {\n      if (!this.rowsPerPage || this.rowsPerPage.length === 0) {\n        this.rowsPerPage = [20, 30, 40];\n      }\n      this.rows = this.rowsPerPage[0];\n    }\n  }\n\n  onChange(event: Event, id: number, key: any) {\n    const target = event.target as HTMLInputElement;\n    this.changeHandler(id, key, target.value);\n  }\n\n  changeHandler(id: number, key: any, value: any) {\n    const column = this.columns.find((item) => item.code === key);\n    if (!this.map.get(id)) {\n      if (column?.type === TableTypeEnum.DATE) {\n        const date = this.parseDate(value);\n        this.dataMap.set(key, date);\n      } else {\n        this.dataMap.set(key, value);\n      }\n      this.map.set(id, new Map(this.dataMap));\n    } else {\n      const mapItem = this.map.get(id) as Map<any, any>;\n      if (column?.type === TableTypeEnum.DATE) {\n        const date = this.parseDate(value);\n        mapItem.set(key, date);\n      } else {\n        mapItem.set(key, value);\n      }\n    }\n  }\n\n  getColumnFilterType(column: TableColumn): string {\n    switch (column.type) {\n      case TableTypeEnum.STRING:\n        return 'text';\n      case TableTypeEnum.AMOUNT:\n      case TableTypeEnum.NUMBER:\n        return 'numeric';\n      case TableTypeEnum.DATE:\n        return 'date';\n      case TableTypeEnum.MULTISELECT:\n        return 'multiSelect';\n      case TableTypeEnum.BOOLEAN:\n        return 'boolean';\n      case TableTypeEnum.COMPOSED:\n        return 'composed';\n      default:\n        return 'text';\n    }\n  }\n\n  isEditable(key: string | undefined): boolean {\n    if (!key) return false;\n    const column = this.columns.find((item) => item.code === key);\n    return column?.isEditable !== false;\n  }\n\n  isMultiSelect(key: any): boolean {\n    const column = this.columns.find((item) => item.code === key);\n    if (\n      column?.type === TableTypeEnum.MULTISELECT &&\n      column.options &&\n      column.code !== undefined\n    ) {\n      this.optionEntries = new Map([\n        [column.code as string, Object.values(column.options)],\n      ]);\n      this.optionValues = this.optionEntries.get(key) || [];\n      return true;\n    }\n    return false;\n  }\n\n  isDatePicker(key: any): boolean {\n    return (\n      this.columns.find((item) => item.code === key)?.type ===\n      TableTypeEnum.DATE\n    );\n  }\n\n  dateConverter(value: any): string {\n    return new Date(value).toLocaleDateString('en-US');\n  }\n\n  getCurrencySymbol(column: TableColumn): string | undefined {\n    return column.type === TableTypeEnum.AMOUNT &&\n      column.currency &&\n      this.isValidCurrencyCode(column.currency)\n      ? column.currency\n      : undefined;\n  }\n\n  private isValidCurrencyCode(currencyCode: string): boolean {\n    return this.validCurrencyCodes.includes(currencyCode);\n  }\n\n  filterGlobal(event: Event): void {\n    const target = event.target as HTMLInputElement;\n    const value = target.value.toLowerCase();\n\n    const filteredData = this.data.filter((item) => {\n      return this.globalFilterFields.some((field) => {\n        const column = this.columns.find((col) => col.code === field);\n        if (!column) {\n          return false;\n        }\n\n        if (column.type === TableTypeEnum.DATE) {\n          const itemDate = this.formatDate(item[field]);\n          return itemDate && itemDate.includes(value);\n        } else if (\n          column.type === TableTypeEnum.AMOUNT ||\n          column.type === TableTypeEnum.NUMBER\n        ) {\n          return (\n            item[field] && item[field].toString().toLowerCase().includes(value)\n          );\n        } else if (column.type === TableTypeEnum.COMPOSED) {\n          return this.filterComposedColumn(item[field], value);\n        } else {\n          return (\n            item[field] && item[field].toString().toLowerCase().includes(value)\n          );\n        }\n      });\n    });\n\n    this.dt.value = filteredData;\n\n    // ⛔ ne pas toucher à totalRecords en mode serveur\n    // this.totalRecords = filteredData.length ?? 0;\n  }\n\n  private filterComposedColumn(composedData: any, value: string): boolean {\n    if (composedData) {\n      return Object.keys(composedData).some((key) => {\n        const cellValue = composedData[key];\n        if (typeof cellValue === 'string') {\n          return cellValue.toLowerCase().includes(value);\n        }\n        return false;\n      });\n    }\n    return false;\n  }\n\n  formatDate(date: any): string {\n    if (!date) return '';\n    if (date instanceof Date) {\n      const day = date.getDate().toString().padStart(2, '0');\n      const month = (date.getMonth() + 1).toString().padStart(2, '0');\n      const year = date.getFullYear().toString();\n      return `${day}/${month}/${year}`;\n    } else if (typeof date === 'string') {\n      const parts = date.split('/');\n      if (parts.length === 3) {\n        return `${parts[0]}/${parts[1]}/${parts[2]}`;\n      } else {\n        return date;\n      }\n    }\n    return '';\n  }\n\n  exportExcel() {\n    this.exportExcelEvent.emit();\n  }\n\n  exportPdf() {\n    this.exportPdfEvent.emit();\n  }\n\n  getImageStyle(style: TitleStyle | ImageStyle | undefined): {\n    [key: string]: string;\n  } {\n    if (style) {\n      const imageStyle: { [key: string]: string } = {\n        width: (style as ImageStyle).width || 'auto',\n        height: (style as ImageStyle).height || 'auto',\n      };\n\n      if ((style as ImageStyle).margin) {\n        imageStyle.margin = (style as ImageStyle).margin!;\n      }\n      if ((style as ImageStyle).marginLeft) {\n        imageStyle.marginLeft = (style as ImageStyle).marginLeft!;\n      }\n      if ((style as ImageStyle).marginRight) {\n        imageStyle.marginRight = (style as ImageStyle).marginRight!;\n      }\n      if ((style as ImageStyle).marginTop) {\n        imageStyle.marginTop = (style as ImageStyle).marginTop!;\n      }\n      if ((style as ImageStyle).marginBottom) {\n        imageStyle.marginBottom = (style as ImageStyle).marginBottom!;\n      }\n\n      return imageStyle;\n    }\n\n    return {};\n  }\n\n  getTitleStyle(style: TitleStyle | ImageStyle | undefined): {\n    [key: string]: string;\n  } {\n    if (style) {\n      return {\n        color: style.color || 'inherit',\n        fontSize: (style as TitleStyle).fontSize || 'inherit',\n        textAlign: (style as TitleStyle).position || 'left',\n      };\n    }\n    return {};\n  }\n\n  formatNumber(\n    value: number,\n    decimalPlaces?: number,\n    thousandSeparator: 'comma' | 'space' = 'comma',\n    decimalSeparator: 'comma' | 'dot' = 'dot'\n  ): string {\n    if (value === null || value === undefined || isNaN(value)) return '';\n\n    let formattedNumber =\n      decimalPlaces !== undefined\n        ? value.toFixed(decimalPlaces)\n        : value.toString();\n\n    if (decimalSeparator === 'comma') {\n      formattedNumber = formattedNumber.replace('.', ',');\n    }\n\n    if (thousandSeparator && Math.abs(value) >= 1000) {\n      const parts = formattedNumber.split(\n        decimalSeparator === 'comma' ? ',' : '.'\n      );\n      parts[0] = parts[0].replace(\n        /\\B(?=(\\d{3})+(?!\\d))/g,\n        thousandSeparator === 'comma' ? ',' : ' '\n      );\n      formattedNumber = parts.join(decimalSeparator === 'comma' ? ',' : '.');\n    }\n\n    return formattedNumber;\n  }\n}\n","<div class=\"pt-advanced-prime-table table-container\">\n  <p-table\n    #dt\n    [value]=\"data\"\n    [loading]=\"loading\"\n    [rows]=\"rows\"\n    [paginator]=\"isPaginated\"\n    [globalFilterFields]=\"globalFilterFields\"\n    [rowsPerPageOptions]=\"rowsPerPage\"\n    [totalRecords]=\"totalRecords\"\n    [lazy]=\"true\"\n    dataKey=\"id\"\n    styleClass=\"p-datatable-gridlines p-datatable-striped\"\n    [scrollable]=\"true\"\n    [scrollHeight]=\"maxHeight !== null ? maxHeight : undefined\"\n    (onFilter)=\"onFilter($event)\"\n    (onPage)=\"onPageChange($event)\"\n  >\n    <!-- CAPTION -->\n    <ng-template pTemplate=\"caption\">\n      <div class=\"flex\">\n        <div>\n          <h3>Total: {{ totalRecords }}</h3>\n        </div>\n\n        <div>\n          <!-- Clear filters -->\n          <button\n            *ngIf=\"hasSearchFilter\"\n            pButton\n            icon=\"pi pi-filter-slash\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"clear(dt)\"\n            title=\"Clear filters\"\n          ></button>\n\n          <!-- Export to Excel Button -->\n          <button\n            *ngIf=\"hasExportExcel\"\n            pButton\n            icon=\"pi pi-file-excel\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"exportExcel()\"\n            title=\"Export to Excel\"\n          ></button>\n\n          <!-- Export to PDF Button -->\n          <button\n            *ngIf=\"hasExportPDF\"\n            pButton\n            icon=\"pi pi-file-pdf\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"exportPdf()\"\n            title=\"Export to PDF\"\n          ></button>\n        </div>\n\n        <div class=\"ml-auto\" *ngIf=\"hasSearchFilter\">\n          <p-iconField iconPosition=\"left\" class=\"ml-auto\">\n            <p-inputIcon>\n              <i class=\"pi pi-search\"></i>\n            </p-inputIcon>\n            <input\n              pInputText\n              type=\"text\"\n              [(ngModel)]=\"searchValue\"\n              (input)=\"filterGlobal($event)\"\n              placeholder=\"Search keyword\"\n            />\n          </p-iconField>\n        </div>\n      </div>\n    </ng-template>\n\n    <!-- HEADER -->\n    <ng-template pTemplate=\"header\">\n      <tr class=\"sticky-header\">\n        <ng-container *ngFor=\"let col of columns\">\n          <th\n            *ngIf=\"!col.children; else groupHeader\"\n            [style.width]=\"getHeaderWidth(col)\"\n            [style.padding]=\"'0px'\"\n            colspan=\"1\"\n          >\n            <ng-container\n              *ngIf=\"isSortable && col.isSortable !== false; else noSortHeader\"\n            >\n              <th\n                pSortableColumn=\"{{ col.code }}\"\n                [style.width]=\"getHeaderWidth(col)\"\n              >\n                <div\n                  class=\"header-container d-flex align-items-center justify-content-between\"\n                  [style.width]=\"col.width\"\n                  [style.padding]=\"'0px'\"\n                  [style.margin]=\"'10px'\"\n                >\n                  <span>{{ col.title }}</span>\n                  <div\n                    class=\"icons d-flex align-items-center\"\n                    [style.width]=\"'77px'\"\n                  >\n                    <p-sortIcon field=\"{{ col.code }}\" />\n                    <ng-container *ngIf=\"col.isFilter !== false\">\n                      <!-- COMPOSED FILTER -->\n                      <p-columnFilter\n                        display=\"menu\"\n                        [field]=\"col.code\"\n                        [type]=\"getColumnFilterType(col)\"\n                        *ngIf=\"col.type === TableTypeEnum.COMPOSED\"\n                        showClearButton=\"false\"\n                        showApplyButton=\"false\"\n                      >\n                        <ng-template\n                          pTemplate=\"filter\"\n                          let-value\n                          let-filterCallback=\"filterCallback\"\n                          (onFilter)=\"onFilter($event)\"\n                        >\n                          <div *ngFor=\"let composedName of col.composedNames\">\n                            <ng-container\n                              *ngIf=\"\n                                getComposedFieldType(col, composedName) ===\n                                TableTypeEnum.STRING\n                              \"\n                            >\n                              <p-multiSelect\n                                [ngModel]=\"filters[composedName]?.value\"\n                                [options]=\"filters[composedName]?.options\"\n                                (onChange)=\"\n                                  onComposedFilterChange(\n                                    col,\n                                    composedName,\n                                    $event.value\n                                  )\n                                \"\n                                [placeholder]=\"\n                                  filters[composedName]?.placeholder\n                                \"\n                                [display]=\"'chip'\"\n                              >\n                                <ng-template let-item pTemplate=\"item\">\n                                  <div class=\"custom-multiselect-item\">\n                                    <img\n                                      *ngIf=\"item.image\"\n                                      [src]=\"item.image\"\n                                      alt=\"icon\"\n                                      class=\"filter-image\"\n                                    />\n                                    <span>{{ item.label }}</span>\n                                  </div>\n                                </ng-template>\n                              </p-multiSelect>\n                            </ng-container>\n                          </div>\n\n                          <ng-template let-item pTemplate=\"item\">\n                            <div class=\"custom-multiselect-item\">\n                              <img\n                                *ngIf=\"item.image\"\n                                [src]=\"item.image\"\n                                alt=\"icon\"\n                                class=\"filter-image\"\n                              />\n                              <span>{{ item.label }}</span>\n                            </div>\n                          </ng-template>\n                        </ng-template>\n                      </p-columnFilter>\n\n                      <!-- OTHER TYPES -->\n                      <p-columnFilter\n                        display=\"menu\"\n                        [field]=\"col.code\"\n                        [type]=\"getColumnFilterType(col)\"\n                        *ngIf=\"col.type !== TableTypeEnum.COMPOSED\"\n                        hideOnClear=\"true\"\n                      >\n                        <!-- NUMBER -->\n                        <ng-template\n                          pTemplate=\"filter\"\n                          let-value\n                          let-filterCallback=\"filterCallback\"\n                          *ngIf=\"col.type === TableTypeEnum.NUMBER\"\n                          (onFilter)=\"onFilter($event)\"\n                        >\n                          <input\n                            pInputText\n                            type=\"number\"\n                            [step]=\"\n                              col.decimalPlaces\n                                ? '0.' + '1'.padEnd(col.decimalPlaces, '0')\n                                : 'any'\n                            \"\n                            [ngModel]=\"value\"\n                            (ngModelChange)=\"filterCallback($event)\"\n                            placeholder=\"Enter a number\"\n                          />\n                        </ng-template>\n\n                        <!-- DATE -->\n                        <ng-template\n                          pTemplate=\"filter\"\n                          let-value\n                          let-filterCallback=\"filterCallback\"\n                          *ngIf=\"col.type === TableTypeEnum.DATE\"\n                          (onFilter)=\"onFilter($event)\"\n                        >\n                          <p-calendar\n                            [ngModel]=\"value\"\n                            (ngModelChange)=\"\n                              onCalendarFilterChange(\n                                $event,\n                                col.code!,\n                                filterCallback\n                              )\n                            \"\n                            dateFormat=\"dd/mm/yy\"\n                            placeholder=\"Choose a date\"\n                          ></p-calendar>\n                        </ng-template>\n\n                        <!-- MULTISELECT -->\n                        <ng-template\n                          pTemplate=\"filter\"\n                          let-value\n                          let-filterCallback=\"filterCallback\"\n                          *ngIf=\"\n                            col.type === TableTypeEnum.MULTISELECT &&\n                            col.filterOptions &&\n                            col.filterOptions.length > 0\n                          \"\n                          (onFilter)=\"onFilter($event)\"\n                        >\n                          <p-multiSelect\n                            [options]=\"col.filterOptions\"\n                            [ngModel]=\"value\"\n                            (ngModelChange)=\"filterCallback($event)\"\n                            display=\"chip\"\n                            placeholder=\"Choose option\"\n                            class=\"custom-multiselect\"\n                          ></p-multiSelect>\n                        </ng-template>\n                      </p-columnFilter>\n                    </ng-container>\n                  </div>\n                </div>\n              </th>\n            </ng-container>\n\n            <ng-template #noSortHeader>\n              <th>\n                <div class=\"header-container\">\n                  <span>{{ col.title }}</span>\n                  <ng-container *ngIf=\"col.isFilter !== false\">\n                    <p-columnFilter\n                      *ngIf=\"col.type === 'AMOUNT'\"\n                      display=\"menu\"\n                      [field]=\"col.code\"\n                      [type]=\"getColumnFilterType(col)\"\n                      [currency]=\"getCurrencySymbol(col)\"\n                    ></p-columnFilter>\n\n                    <p-columnFilter\n                      *ngIf=\"col.type !== 'AMOUNT'\"\n                      display=\"menu\"\n                      [field]=\"col.code\"\n                      [type]=\"getColumnFilterType(col)\"\n                    >\n                      <ng-template\n                        pTemplate=\"filter\"\n                        let-value\n                        let-filterCallback=\"filterCallback\"\n                        *ngIf=\"getColumnFilterType(col) === 'date'\"\n                      >\n                        <p-calendar\n                          [ngModel]=\"value\"\n                          (ngModelChange)=\"filterCallback($event)\"\n                          dateFormat=\"dd/mm/yy\"\n                        ></p-calendar>\n                      </ng-template>\n\n                      <ng-template\n                        pTemplate=\"filter\"\n                        let-value\n                        let-filterCallback=\"filterCallback\"\n                        *ngIf=\"getColumnFilterType(col) === 'multiSelect'\"\n                      >\n                        <p-multiSelect\n                          [options]=\"col.filterOptions\"\n                          [ngModel]=\"value\"\n                          (ngModelChange)=\"filterCallback($event)\"\n                          display=\"chip\"\n                          placeholder=\"Select\"\n                          class=\"custom-multiselect\"\n                        ></p-multiSelect>\n                      </ng-template>\n                    </p-columnFilter>\n                  </ng-container>\n                </div>\n              </th>\n            </ng-template>\n          </th>\n\n          <!-- GROUPED HEADER -->\n          <ng-template #groupHeader>\n            <th\n              [attr.colspan]=\"col.children?.length\"\n              [style.width]=\"getHeaderWidth(col)\"\n              [style.text-align]=\"'center'\"\n            >\n              <span>{{ col.title }}</span>\n            </th>\n          </ng-template>\n        </ng-container>\n      </tr>\n\n      <!-- CHILD HEADERS -->\n      <tr *ngIf=\"hasGroupedColumns\">\n        <ng-container *ngFor=\"let col of columns\">\n          <ng-container *ngIf=\"col.children\">\n            <th\n              *ngFor=\"let child of col.children\"\n              [style.width]=\"getHeaderWidth(child)\"\n              [style.padding]=\"'0px'\"\n            ></th>\n          </ng-container>\n        </ng-container>\n      </tr>\n    </ng-template>\n\n    <!-- EMPTY MESSAGE -->\n    <ng-template pTemplate=\"emptymessage\">\n      <div class=\"empty-message\">\n        <i class=\"pi pi-info-circle\"></i>\n        <p>No records available to display.</p>\n      </div>\n    </ng-template>\n\n    <!-- BODY -->\n    <ng-template\n      pTemplate=\"body\"\n      let-data\n      let-editing=\"editing\"\n      let-ri=\"rowIndex\"\n    >\n      <tr *ngIf=\"!loading\" [pEditableRow]=\"isEdit ? data : null\">\n        <ng-container *ngFor=\"let col of columns\">\n          <ng-container *ngIf=\"!col.children; else childColumns\">\n            <!-- show cell when:\n                 - ACTION column (no value needed)\n                 - OR normal col with value present -->\n            <ng-container\n              *ngIf=\"\n                col.type === TableTypeEnum.ACTION ||\n                (col.code !== undefined && data[col.code!] !== undefined)\n              \"\n            >\n              <td\n                *ngIf=\"\n                  isEditable(col.code!) && col.type !== TableTypeEnum.ACTION;\n                  else normalTD\n                \"\n                [style.width]=\"getHeaderWidth(col)\"\n              >\n                <!-- Editable input for NUMBER/DATE/STRING/MULTISELECT -->\n                <ng-container *ngIf=\"isMultiSelect(col.code); else datePicker\">\n                  <p-cellEditor>\n                    <ng-template pTemplate=\"input\">\n                      <p-multiSelect\n                        appendTo=\"body\"\n                        [ngModel]=\"data[col.code!]\"\n                        [style]=\"{ width: '100%' }\"\n                        (ngModelChange)=\"\n                          changeHandler(data.id, col.code, $event)\n                        \"\n                        [options]=\"optionValues\"\n                      ></p-multiSelect>\n                    </ng-template>\n                    <ng-template pTemplate=\"output\">\n                      <div class=\"multi-select-container\">\n                        <ng-container *ngFor=\"let rec of data[col.code!]\">\n                          <p-tag [value]=\"rec\"></p-tag>\n                        </ng-container>\n                      </div>\n                    </ng-template>\n                  </p-cellEditor>\n                </ng-container>\n\n                <ng-template #datePicker>\n                  <ng-container\n                    *ngIf=\"isDatePicker(col.code); else normalInput\"\n                  >\n                    <p-cellEditor>\n                      <ng-template pTemplate=\"input\">\n                        <p-calendar\n                          [inputId]=\"data[col.code!]\"\n                          [ngModel]=\"data[col.code!]\"\n                          (ngModelChange)=\"\n                            changeHandler(data.id, col.code, $event)\n                          \"\n                          [dateFormat]=\"'dd/mm/yy'\"\n                        ></p-calendar>\n                      </ng-template>\n                      <ng-template pTemplate=\"output\">\n                        {{ data[col.code!] | customDate }}\n                      </ng-template>\n                    </p-cellEditor>\n                  </ng-container>\n                </ng-template>\n\n                <ng-template #normalInput>\n                  <p-cellEditor>\n                    <ng-template pTemplate=\"input\">\n                      <input\n                        pInputText\n                        type=\"text\"\n                        [ngModel]=\"data[col.code!]\"\n                        (change)=\"onChange($event, data.id, col.code)\"\n                      />\n                    </ng-template>\n                    <ng-template pTemplate=\"output\">\n                      <ng-container\n                        *ngIf=\"\n                          col.type === TableTypeEnum.AMOUNT;\n                          else normalOutput\n                        \"\n                      >\n                        {{\n                          data[col.code!]\n                            | customCurrency\n                              : getCurrencySymbol(col)\n                              : col.decimalPlaces\n                              : col.thousandSeparator\n                              : col.decimalSeparator\n                        }}\n                      </ng-container>\n                      <ng-template #normalOutput>\n                        {{ data[col.code!] }}\n                      </ng-template>\n                    </ng-template>\n                  </p-cellEditor>\n                </ng-template>\n              </td>\n\n              <ng-template #normalTD>\n                <td [style.width]=\"getHeaderWidth(col)\">\n                  <!-- ACTION column: built-in edit/delete + custom actions -->\n                  <ng-container\n                    *ngIf=\"\n                      col.type === TableTypeEnum.ACTION;\n                      else nonActionCell\n                    \"\n                  >\n                    <div class=\"action-buttons-container\">\n                      <!-- built-in delete -->\n                      <button\n                        *ngIf=\"isDelete\"\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        icon=\"pi pi-trash\"\n                        (click)=\"Delete(data.id)\"\n                        class=\"p-button-rounded p-button-text\"\n                      ></button>\n\n                      <!-- built-in inline edit (old behaviour) -->\n                      <div *ngIf=\"isEdit\">\n                        <button\n                          pInitEditableRow\n                          *ngIf=\"!editing\"\n                          pButton\n                          pRipple\n                          type=\"button\"\n                          icon=\"pi pi-pencil\"\n                          (click)=\"initEditableRow(data)\"\n                          class=\"p-button-rounded p-button-text\"\n                        ></button>\n                        <button\n                          *ngIf=\"editing\"\n                          pSaveEditableRow\n                          pButton\n                          pRipple\n                          type=\"button\"\n                          icon=\"pi pi-check\"\n                          (click)=\"saveEditableRow(data)\"\n                          class=\"p-button-rounded p-button-text\"\n                        ></button>\n                        <button\n                          *ngIf=\"editing\"\n                          pCancelEditableRow\n                          pButton\n                          pRipple\n                          type=\"button\"\n                          icon=\"pi pi-times\"\n                          (click)=\"cancelEditableRow(data)\"\n                          class=\"p-button-rounded p-button-text\"\n                        ></button>\n                      </div>\n\n                      <!-- custom actions -->\n                      <button\n                        *ngFor=\"let act of customActions\"\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        class=\"p-button-rounded p-button-text\"\n                        [icon]=\"act.icon || 'pi pi-ellipsis-h'\"\n                        [ngClass]=\"act.styleClass\"\n                        (click)=\"onCustomActionClick(act, data)\"\n                      ></button>\n                    </div>\n                  </ng-container>\n\n                  <!-- NON-ACTION cells -->\n                  <ng-template #nonActionCell>\n                    <!-- COMPOSED -->\n                    <ng-container\n                      *ngIf=\"\n                        col.type === TableTypeEnum.COMPOSED;\n                        else nonComposed\n                      \"\n                    >\n                      <div class=\"composed-cell\">\n                        <ng-container\n                          *ngFor=\"\n                            let composedName of col.composedNames;\n                            let i = index\n                          \"\n                        >\n                          <!-- IMAGE -->\n                          <ng-container\n                            *ngIf=\"\n                              col.composedTypes &&\n                              col.composedTypes[i] === TableTypeEnum.IMAGE\n                            \"\n                          >\n                            <img\n                              [src]=\"data[col.code!][composedName]\"\n                              alt=\"composed-img\"\n                              class=\"composed-image\"\n                              [ngStyle]=\"\n                                getImageStyle(\n                                  col.composedStyles?.[composedName]\n                                )\n                              \"\n                            />\n                          </ng-container>\n\n                          <!-- STRING -->\n                          <ng-container\n                            *ngIf=\"\n                              col.composedTypes &&\n                              col.composedTypes[i] === TableTypeEnum.STRING\n                            \"\n                          >\n                            <span\n                              class=\"composed-text\"\n                              [ngStyle]=\"\n                                getTitleStyle(\n                                  col.composedStyles?.[composedName]\n                                )\n                              \"\n                            >\n                              {{ data[col.code!][composedName] }}\n                            </span>\n                          </ng-container>\n                        </ng-container>\n                      </div>\n                    </ng-container>\n\n                    <ng-template #nonComposed>\n                      <!-- AMOUNT-->\n                      <ng-container\n                        *ngIf=\"\n                          col.type === TableTypeEnum.AMOUNT;\n                          else nonAmount\n                        \"\n                      >\n                        {{\n                          data[col.code!]\n                            | customCurrency\n                              : getCurrencySymbol(col)\n                              : col.decimalPlaces\n                              : col.thousandSeparator\n                              : col.decimalSeparator\n                        }}\n                      </ng-container>\n\n                      <ng-template #nonAmount>\n                        <!-- NUMBER-->\n                        <ng-container\n                          *ngIf=\"\n                            col.type === TableTypeEnum.NUMBER;\n                            else nonNumber\n                          \"\n                        >\n                          {{\n                            formatNumber(\n                              data[col.code!],\n                              col.decimalPlaces,\n                              col.thousandSeparator,\n                              col.decimalSeparator\n                            )\n                          }}\n                        </ng-container>\n\n                        <ng-template #nonNumber>\n                          <!-- DATE -->\n                          <ng-container\n                            *ngIf=\"\n                              col.type === TableTypeEnum.DATE;\n                              else normalTypes\n                            \"\n                          >\n                            {{ formatDate(data[col.code!]) }}\n                          </ng-container>\n\n                          <ng-template #normalTypes>\n                            <!-- STRING, MULTISELECT-->\n                            <ng-container\n                              *ngIf=\"\n                                [\n                                  TableTypeEnum.STRING,\n                                  TableTypeEnum.MULTISELECT\n                                ].includes(col.type!)\n                              \"\n                            >\n                              {{ data[col.code!] }}\n                            </ng-container>\n                          </ng-template>\n                        </ng-template>\n                      </ng-template>\n                    </ng-template>\n                  </ng-template>\n                </td>\n              </ng-template>\n            </ng-container>\n          </ng-container>\n\n          <!-- CHILD COLUMNS -->\n          <ng-template #childColumns>\n            <ng-container *ngFor=\"let child of col.children\">\n              <ng-container\n                *ngIf=\"\n                  child.code !== undefined && data[child.code] !== undefined\n                \"\n              >\n                <td [style.width]=\"getHeaderWidth(child)\">\n                  <ng-container\n                    *ngIf=\"isEditable(child.code); else childNormalTD\"\n                  >\n                    <p-cellEditor>\n                      <ng-template pTemplate=\"input\">\n                        <input\n                          pInputText\n                          type=\"text\"\n                          [ngModel]=\"data[child.code]\"\n                          (change)=\"onChange($event, data.id, child.code)\"\n                        />\n                      </ng-template>\n                      <ng-template pTemplate=\"output\">\n                        {{ data[child.code] }}\n                      </ng-template>\n                    </p-cellEditor>\n                  </ng-container>\n\n                  <ng-template #childNormalTD>\n                    {{ data[child.code] }}\n                  </ng-template>\n                </td>\n              </ng-container>\n            </ng-container>\n          </ng-template>\n        </ng-container>\n      </tr>\n    </ng-template>\n  </p-table>\n</div>\n"]}
|
|
@@ -287,7 +287,10 @@ class PTAdvancedPrimeTableComponent {
|
|
|
287
287
|
/** Called when a custom action button is clicked */
|
|
288
288
|
onCustomActionClick(action, row) {
|
|
289
289
|
if (action && typeof action.action === 'function') {
|
|
290
|
-
|
|
290
|
+
// Defer to next macrotask → fresh change detection cycle
|
|
291
|
+
setTimeout(() => {
|
|
292
|
+
action.action(row);
|
|
293
|
+
}, 0);
|
|
291
294
|
}
|
|
292
295
|
}
|
|
293
296
|
// Initialize filters for composed columns
|