@sarasanalytics-com/design-system 0.0.166 → 0.0.168

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (91) hide show
  1. package/README.md +24 -24
  2. package/esm2022/interfaces/avatar-interface.mjs +1 -1
  3. package/esm2022/interfaces/button-interface.mjs +1 -1
  4. package/esm2022/interfaces/card-carousel-interface.mjs +1 -1
  5. package/esm2022/interfaces/chip-interface.mjs +1 -1
  6. package/esm2022/interfaces/form-layout.interface.mjs +1 -1
  7. package/esm2022/interfaces/grid-interface.mjs +1 -1
  8. package/esm2022/interfaces/guide-card-interface.mjs +1 -1
  9. package/esm2022/interfaces/header-interface.mjs +1 -1
  10. package/esm2022/interfaces/icon-interface.mjs +1 -1
  11. package/esm2022/interfaces/layout-section-interface.mjs +1 -1
  12. package/esm2022/interfaces/left-nav-interface.mjs +1 -1
  13. package/esm2022/interfaces/menu.interface.mjs +1 -1
  14. package/esm2022/interfaces/message-banner.mjs +1 -1
  15. package/esm2022/interfaces/option-interface.mjs +1 -1
  16. package/esm2022/interfaces/scrolling-card-interface.mjs +1 -1
  17. package/esm2022/interfaces/select-interface.mjs +1 -1
  18. package/esm2022/interfaces/status-dot.interface.mjs +1 -1
  19. package/esm2022/interfaces/tab-interface.mjs +1 -1
  20. package/esm2022/interfaces/toast-interface.mjs +1 -1
  21. package/esm2022/interfaces/typography-animation-interface.mjs +1 -1
  22. package/esm2022/lib/accordion/accordion.component.mjs +3 -3
  23. package/esm2022/lib/avatar/avatar.component.mjs +3 -3
  24. package/esm2022/lib/button/button.component.mjs +3 -3
  25. package/esm2022/lib/calendar-header/calendar-header.component.mjs +3 -3
  26. package/esm2022/lib/card/card-body/card-body.component.mjs +1 -1
  27. package/esm2022/lib/card/card-custom-header/card-custom-header.component.mjs +1 -1
  28. package/esm2022/lib/card/card-footer-actions/card-footer-actions.component.mjs +3 -3
  29. package/esm2022/lib/card/card-icon/card-icon.component.mjs +1 -1
  30. package/esm2022/lib/card/card-title-actions/card-title-actions.component.mjs +1 -1
  31. package/esm2022/lib/card/card.component.mjs +3 -3
  32. package/esm2022/lib/card/checkbox-card/checkbox-card.component.mjs +3 -3
  33. package/esm2022/lib/card/guide-card/guide-card.component.mjs +3 -3
  34. package/esm2022/lib/card/menu-card/menu-card.component.mjs +3 -3
  35. package/esm2022/lib/card/thumbnail-card/thumbnail-card.component.mjs +3 -3
  36. package/esm2022/lib/card-carousel/card-carousel.component.mjs +3 -3
  37. package/esm2022/lib/categories-nav/categories-nav.component.mjs +3 -3
  38. package/esm2022/lib/checkbox/checkbox.component.mjs +3 -3
  39. package/esm2022/lib/chips/chips.component.mjs +3 -3
  40. package/esm2022/lib/component-library.component.mjs +9 -9
  41. package/esm2022/lib/component-library.service.mjs +1 -1
  42. package/esm2022/lib/dashboard-loader/dashboard-loader.component.mjs +1 -1
  43. package/esm2022/lib/data-grid/data-grid.component.mjs +3 -3
  44. package/esm2022/lib/datepicker/datepicker.component.mjs +3 -3
  45. package/esm2022/lib/dialog/dialog.component.mjs +3 -3
  46. package/esm2022/lib/dropdown/category-dropdown/category-dropdown.component.mjs +3 -3
  47. package/esm2022/lib/dropdown/ng-select/ng-select.component.mjs +3 -3
  48. package/esm2022/lib/filter/filter.component.mjs +3 -3
  49. package/esm2022/lib/form-input/form-input.component.mjs +4 -4
  50. package/esm2022/lib/form-select/form-select.component.mjs +3 -3
  51. package/esm2022/lib/grid-cell/grid-cell.component.mjs +10 -10
  52. package/esm2022/lib/header/header.component.mjs +3 -3
  53. package/esm2022/lib/icon/icon.component.mjs +3 -3
  54. package/esm2022/lib/icon/icon.service.mjs +1 -1
  55. package/esm2022/lib/layout-section/layout-section.component.mjs +3 -3
  56. package/esm2022/lib/left-nav/left-nav.component.mjs +3 -3
  57. package/esm2022/lib/list/list.component.mjs +3 -3
  58. package/esm2022/lib/menu/menu-list/menu-item.component.mjs +3 -3
  59. package/esm2022/lib/menu/menu.component.mjs +3 -3
  60. package/esm2022/lib/menu/menu.directive.mjs +1 -1
  61. package/esm2022/lib/message-banner/message-banner.component.mjs +3 -3
  62. package/esm2022/lib/message-banner-v2/message-banner-v2.component.mjs +3 -3
  63. package/esm2022/lib/mini-card/mini-card.component.mjs +3 -3
  64. package/esm2022/lib/page-layout/page-layout.component.mjs +3 -3
  65. package/esm2022/lib/progress-bar/progress-bar.component.mjs +3 -3
  66. package/esm2022/lib/query-builder/query-builder-demo.component.mjs +45 -45
  67. package/esm2022/lib/query-builder/query-builder.component.mjs +3 -3
  68. package/esm2022/lib/query-builder/query-builder.service.mjs +1 -1
  69. package/esm2022/lib/query-builder-formly/query-builder-formly.component.mjs +128 -136
  70. package/esm2022/lib/query-builder-textarea/query-builder-textarea-demo.component.mjs +39 -39
  71. package/esm2022/lib/query-builder-textarea/query-builder-textarea.component.mjs +3 -3
  72. package/esm2022/lib/radio-button/radio-button.component.mjs +3 -3
  73. package/esm2022/lib/scrolling-cards/scrolling-cards.component.mjs +3 -3
  74. package/esm2022/lib/skeleton/skeleton-base.component.mjs +1 -1
  75. package/esm2022/lib/skeleton/skeleton-container.component.mjs +9 -9
  76. package/esm2022/lib/skeleton/skeleton-loader.component.mjs +217 -217
  77. package/esm2022/lib/skeleton/skeleton-presets.mjs +1 -1
  78. package/esm2022/lib/skeleton/skeleton-shapes.component.mjs +1 -1
  79. package/esm2022/lib/spinner/spinner.component.mjs +1 -1
  80. package/esm2022/lib/status-dot/status-dot.component.mjs +17 -17
  81. package/esm2022/lib/stepper/stepper.component.mjs +3 -3
  82. package/esm2022/lib/tabs/tabs.component.mjs +3 -3
  83. package/esm2022/lib/toast/toast.component.mjs +3 -3
  84. package/esm2022/lib/tool-tip/tool-tip.component.mjs +3 -3
  85. package/esm2022/lib/typography-animation/typography-animation.component.mjs +3 -3
  86. package/esm2022/utils/validators.mjs +1 -1
  87. package/fesm2022/sarasanalytics-com-design-system.mjs +557 -563
  88. package/fesm2022/sarasanalytics-com-design-system.mjs.map +1 -1
  89. package/lib/query-builder-formly/query-builder-formly.component.d.ts +15 -3
  90. package/package.json +1 -1
  91. package/styles/styles.css +434 -434
@@ -104,4 +104,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImpor
104
104
  providedIn: 'root'
105
105
  }]
106
106
  }], ctorParameters: () => [] });
107
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicXVlcnktYnVpbGRlci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50LWxpYnJhcnkvc3JjL2xpYi9xdWVyeS1idWlsZGVyL3F1ZXJ5LWJ1aWxkZXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQU0zQyxNQUFNLE9BQU8sbUJBQW1CO0lBRTlCLGdCQUFnQixDQUFDO0lBRWpCOzs7O09BSUc7SUFDSCxhQUFhLENBQUMsS0FBaUI7UUFDN0IsSUFBSSxDQUFDLEtBQUssSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDdkQsT0FBTyxFQUFFLENBQUM7UUFDWixDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUMsS0FBSzthQUNmLEdBQUcsQ0FBQyxDQUFDLElBQVMsRUFBRSxFQUFFO1lBQ2pCLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUNuQixPQUFPLElBQUksSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFrQixDQUFDLEdBQUcsQ0FBQztZQUN2RCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sT0FBTyxHQUFHLElBQUksQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDeEQsQ0FBQztRQUNILENBQUMsQ0FBQzthQUNELElBQUksQ0FBQyxJQUFJLEtBQUssQ0FBQyxTQUFTLENBQUMsV0FBVyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ2hELENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsV0FBVyxDQUFDLEtBQWlCO1FBQzNCLElBQUksQ0FBQyxLQUFLLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxJQUFJLEtBQUssQ0FBQyxLQUFLLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQ3ZELE9BQU8sRUFBRSxDQUFDO1FBQ1osQ0FBQztRQUVELE9BQU87WUFDTCxTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVM7WUFDMUIsS0FBSyxFQUFFLEtBQUssQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBUyxFQUFFLEVBQUU7Z0JBQ25DLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO29CQUNuQixPQUFPLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBa0IsQ0FBQyxDQUFDO2dCQUM5QyxDQUFDO3FCQUFNLENBQUM7b0JBQ04sT0FBTzt3QkFDTCxLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUs7d0JBQ2pCLFFBQVEsRUFBRSxJQUFJLENBQUMsUUFBUTt3QkFDdkIsS0FBSyxFQUFFLElBQUksQ0FBQyxLQUFLO3FCQUNsQixDQUFDO2dCQUNKLENBQUM7WUFDSCxDQUFDLENBQUM7U0FDSCxDQUFDO0lBQ0osQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxZQUFZLENBQUMsS0FBaUI7UUFDNUIsSUFBSSxDQUFDLEtBQUssSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDdkQsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDLElBQVMsRUFBRSxFQUFFO1lBQ3JDLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUNuQixPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBa0IsQ0FBQyxDQUFDO1lBQy9DLENBQUM7aUJBQU0sQ0FBQztnQkFDTixPQUFPLElBQUksQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLFNBQVMsSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLElBQUksQ0FBQztZQUN4RixDQUFDO1FBQ0gsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILGdCQUFnQixDQUFDLFlBQTBCLEtBQUs7UUFDOUMsT0FBTztZQUNMLFNBQVM7WUFDVCxLQUFLLEVBQUUsRUFBRTtTQUNWLENBQUM7SUFDSixDQUFDO0lBRUQ7Ozs7OztPQU1HO0lBQ0gsaUJBQWlCLENBQUMsS0FBYSxFQUFFLFFBQWdCLEVBQUUsS0FBVTtRQUMzRCxPQUFPO1lBQ0wsU0FBUyxFQUFFLEtBQUs7WUFDaEIsS0FBSyxFQUFFO2dCQUNMO29CQUNFLEtBQUs7b0JBQ0wsUUFBUTtvQkFDUixLQUFLO2lCQUNOO2FBQ0Y7U0FDRixDQUFDO0lBQ0osQ0FBQzs4R0FwR1UsbUJBQW1CO2tIQUFuQixtQkFBbUIsY0FGbEIsTUFBTTs7MkZBRVAsbUJBQW1CO2tCQUgvQixVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgUXVlcnlHcm91cCwgUXVlcnlSdWxlIH0gZnJvbSAnLi9xdWVyeS1idWlsZGVyLmNvbXBvbmVudCc7XHJcblxyXG5ASW5qZWN0YWJsZSh7XHJcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBRdWVyeUJ1aWxkZXJTZXJ2aWNlIHtcclxuICBcclxuICBjb25zdHJ1Y3RvcigpIHsgfVxyXG5cclxuICAvKipcclxuICAgKiBDb252ZXJ0cyBhIHF1ZXJ5IGdyb3VwIHRvIGEgaHVtYW4tcmVhZGFibGUgc3RyaW5nXHJcbiAgICogQHBhcmFtIHF1ZXJ5IFRoZSBxdWVyeSBncm91cCB0byBjb252ZXJ0XHJcbiAgICogQHJldHVybnMgQSBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgdGhlIHF1ZXJ5XHJcbiAgICovXHJcbiAgcXVlcnlUb1N0cmluZyhxdWVyeTogUXVlcnlHcm91cCk6IHN0cmluZyB7XHJcbiAgICBpZiAoIXF1ZXJ5IHx8ICFxdWVyeS5ydWxlcyB8fCBxdWVyeS5ydWxlcy5sZW5ndGggPT09IDApIHtcclxuICAgICAgcmV0dXJuICcnO1xyXG4gICAgfVxyXG5cclxuICAgIHJldHVybiBxdWVyeS5ydWxlc1xyXG4gICAgICAubWFwKChydWxlOiBhbnkpID0+IHtcclxuICAgICAgICBpZiAocnVsZS5jb25kaXRpb24pIHtcclxuICAgICAgICAgIHJldHVybiBgKCR7dGhpcy5xdWVyeVRvU3RyaW5nKHJ1bGUgYXMgUXVlcnlHcm91cCl9KWA7XHJcbiAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgIHJldHVybiBgJHtydWxlLmZpZWxkfSAke3J1bGUub3BlcmF0b3J9ICR7cnVsZS52YWx1ZX1gO1xyXG4gICAgICAgIH1cclxuICAgICAgfSlcclxuICAgICAgLmpvaW4oYCAke3F1ZXJ5LmNvbmRpdGlvbi50b1VwcGVyQ2FzZSgpfSBgKTtcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIENvbnZlcnRzIGEgcXVlcnkgZ3JvdXAgdG8gYSBKU09OIG9iamVjdCB0aGF0IGNhbiBiZSB1c2VkIGluIEFQSSByZXF1ZXN0c1xyXG4gICAqIEBwYXJhbSBxdWVyeSBUaGUgcXVlcnkgZ3JvdXAgdG8gY29udmVydFxyXG4gICAqIEByZXR1cm5zIEEgSlNPTiBvYmplY3QgcmVwcmVzZW50YXRpb24gb2YgdGhlIHF1ZXJ5XHJcbiAgICovXHJcbiAgcXVlcnlUb0pzb24ocXVlcnk6IFF1ZXJ5R3JvdXApOiBhbnkge1xyXG4gICAgaWYgKCFxdWVyeSB8fCAhcXVlcnkucnVsZXMgfHwgcXVlcnkucnVsZXMubGVuZ3RoID09PSAwKSB7XHJcbiAgICAgIHJldHVybiB7fTtcclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4ge1xyXG4gICAgICBjb25kaXRpb246IHF1ZXJ5LmNvbmRpdGlvbixcclxuICAgICAgcnVsZXM6IHF1ZXJ5LnJ1bGVzLm1hcCgocnVsZTogYW55KSA9PiB7XHJcbiAgICAgICAgaWYgKHJ1bGUuY29uZGl0aW9uKSB7XHJcbiAgICAgICAgICByZXR1cm4gdGhpcy5xdWVyeVRvSnNvbihydWxlIGFzIFF1ZXJ5R3JvdXApO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICByZXR1cm4ge1xyXG4gICAgICAgICAgICBmaWVsZDogcnVsZS5maWVsZCxcclxuICAgICAgICAgICAgb3BlcmF0b3I6IHJ1bGUub3BlcmF0b3IsXHJcbiAgICAgICAgICAgIHZhbHVlOiBydWxlLnZhbHVlXHJcbiAgICAgICAgICB9O1xyXG4gICAgICAgIH1cclxuICAgICAgfSlcclxuICAgIH07XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBWYWxpZGF0ZXMgYSBxdWVyeSBncm91cCB0byBlbnN1cmUgYWxsIHJ1bGVzIGhhdmUgdmFsaWQgdmFsdWVzXHJcbiAgICogQHBhcmFtIHF1ZXJ5IFRoZSBxdWVyeSBncm91cCB0byB2YWxpZGF0ZVxyXG4gICAqIEByZXR1cm5zIFRydWUgaWYgdGhlIHF1ZXJ5IGlzIHZhbGlkLCBmYWxzZSBvdGhlcndpc2VcclxuICAgKi9cclxuICBpc1F1ZXJ5VmFsaWQocXVlcnk6IFF1ZXJ5R3JvdXApOiBib29sZWFuIHtcclxuICAgIGlmICghcXVlcnkgfHwgIXF1ZXJ5LnJ1bGVzIHx8IHF1ZXJ5LnJ1bGVzLmxlbmd0aCA9PT0gMCkge1xyXG4gICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICB9XHJcblxyXG4gICAgcmV0dXJuIHF1ZXJ5LnJ1bGVzLmV2ZXJ5KChydWxlOiBhbnkpID0+IHtcclxuICAgICAgaWYgKHJ1bGUuY29uZGl0aW9uKSB7XHJcbiAgICAgICAgcmV0dXJuIHRoaXMuaXNRdWVyeVZhbGlkKHJ1bGUgYXMgUXVlcnlHcm91cCk7XHJcbiAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgcmV0dXJuIHJ1bGUuZmllbGQgJiYgcnVsZS5vcGVyYXRvciAmJiBydWxlLnZhbHVlICE9PSB1bmRlZmluZWQgJiYgcnVsZS52YWx1ZSAhPT0gbnVsbDtcclxuICAgICAgfVxyXG4gICAgfSk7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBDcmVhdGVzIGFuIGVtcHR5IHF1ZXJ5IGdyb3VwXHJcbiAgICogQHBhcmFtIGNvbmRpdGlvbiBUaGUgY29uZGl0aW9uIHRvIHVzZSAoYW5kL29yKVxyXG4gICAqIEByZXR1cm5zIEFuIGVtcHR5IHF1ZXJ5IGdyb3VwXHJcbiAgICovXHJcbiAgY3JlYXRlRW1wdHlRdWVyeShjb25kaXRpb246ICdhbmQnIHwgJ29yJyA9ICdhbmQnKTogUXVlcnlHcm91cCB7XHJcbiAgICByZXR1cm4ge1xyXG4gICAgICBjb25kaXRpb24sXHJcbiAgICAgIHJ1bGVzOiBbXVxyXG4gICAgfTtcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIENyZWF0ZXMgYSBzaW1wbGUgcXVlcnkgd2l0aCBhIHNpbmdsZSBydWxlXHJcbiAgICogQHBhcmFtIGZpZWxkIFRoZSBmaWVsZCB0byBmaWx0ZXIgb25cclxuICAgKiBAcGFyYW0gb3BlcmF0b3IgVGhlIG9wZXJhdG9yIHRvIHVzZVxyXG4gICAqIEBwYXJhbSB2YWx1ZSBUaGUgdmFsdWUgdG8gZmlsdGVyIGJ5XHJcbiAgICogQHJldHVybnMgQSBxdWVyeSBncm91cCB3aXRoIGEgc2luZ2xlIHJ1bGVcclxuICAgKi9cclxuICBjcmVhdGVTaW1wbGVRdWVyeShmaWVsZDogc3RyaW5nLCBvcGVyYXRvcjogc3RyaW5nLCB2YWx1ZTogYW55KTogUXVlcnlHcm91cCB7XHJcbiAgICByZXR1cm4ge1xyXG4gICAgICBjb25kaXRpb246ICdhbmQnLFxyXG4gICAgICBydWxlczogW1xyXG4gICAgICAgIHtcclxuICAgICAgICAgIGZpZWxkLFxyXG4gICAgICAgICAgb3BlcmF0b3IsXHJcbiAgICAgICAgICB2YWx1ZVxyXG4gICAgICAgIH1cclxuICAgICAgXVxyXG4gICAgfTtcclxuICB9XHJcbn1cclxuIl19
107
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicXVlcnktYnVpbGRlci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50LWxpYnJhcnkvc3JjL2xpYi9xdWVyeS1idWlsZGVyL3F1ZXJ5LWJ1aWxkZXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQU0zQyxNQUFNLE9BQU8sbUJBQW1CO0lBRTlCLGdCQUFnQixDQUFDO0lBRWpCOzs7O09BSUc7SUFDSCxhQUFhLENBQUMsS0FBaUI7UUFDN0IsSUFBSSxDQUFDLEtBQUssSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDdkQsT0FBTyxFQUFFLENBQUM7UUFDWixDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUMsS0FBSzthQUNmLEdBQUcsQ0FBQyxDQUFDLElBQVMsRUFBRSxFQUFFO1lBQ2pCLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUNuQixPQUFPLElBQUksSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFrQixDQUFDLEdBQUcsQ0FBQztZQUN2RCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sT0FBTyxHQUFHLElBQUksQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDeEQsQ0FBQztRQUNILENBQUMsQ0FBQzthQUNELElBQUksQ0FBQyxJQUFJLEtBQUssQ0FBQyxTQUFTLENBQUMsV0FBVyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ2hELENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsV0FBVyxDQUFDLEtBQWlCO1FBQzNCLElBQUksQ0FBQyxLQUFLLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxJQUFJLEtBQUssQ0FBQyxLQUFLLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQ3ZELE9BQU8sRUFBRSxDQUFDO1FBQ1osQ0FBQztRQUVELE9BQU87WUFDTCxTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVM7WUFDMUIsS0FBSyxFQUFFLEtBQUssQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBUyxFQUFFLEVBQUU7Z0JBQ25DLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO29CQUNuQixPQUFPLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBa0IsQ0FBQyxDQUFDO2dCQUM5QyxDQUFDO3FCQUFNLENBQUM7b0JBQ04sT0FBTzt3QkFDTCxLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUs7d0JBQ2pCLFFBQVEsRUFBRSxJQUFJLENBQUMsUUFBUTt3QkFDdkIsS0FBSyxFQUFFLElBQUksQ0FBQyxLQUFLO3FCQUNsQixDQUFDO2dCQUNKLENBQUM7WUFDSCxDQUFDLENBQUM7U0FDSCxDQUFDO0lBQ0osQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxZQUFZLENBQUMsS0FBaUI7UUFDNUIsSUFBSSxDQUFDLEtBQUssSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDdkQsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDLElBQVMsRUFBRSxFQUFFO1lBQ3JDLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUNuQixPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBa0IsQ0FBQyxDQUFDO1lBQy9DLENBQUM7aUJBQU0sQ0FBQztnQkFDTixPQUFPLElBQUksQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLFNBQVMsSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLElBQUksQ0FBQztZQUN4RixDQUFDO1FBQ0gsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILGdCQUFnQixDQUFDLFlBQTBCLEtBQUs7UUFDOUMsT0FBTztZQUNMLFNBQVM7WUFDVCxLQUFLLEVBQUUsRUFBRTtTQUNWLENBQUM7SUFDSixDQUFDO0lBRUQ7Ozs7OztPQU1HO0lBQ0gsaUJBQWlCLENBQUMsS0FBYSxFQUFFLFFBQWdCLEVBQUUsS0FBVTtRQUMzRCxPQUFPO1lBQ0wsU0FBUyxFQUFFLEtBQUs7WUFDaEIsS0FBSyxFQUFFO2dCQUNMO29CQUNFLEtBQUs7b0JBQ0wsUUFBUTtvQkFDUixLQUFLO2lCQUNOO2FBQ0Y7U0FDRixDQUFDO0lBQ0osQ0FBQzs4R0FwR1UsbUJBQW1CO2tIQUFuQixtQkFBbUIsY0FGbEIsTUFBTTs7MkZBRVAsbUJBQW1CO2tCQUgvQixVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFF1ZXJ5R3JvdXAsIFF1ZXJ5UnVsZSB9IGZyb20gJy4vcXVlcnktYnVpbGRlci5jb21wb25lbnQnO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290J1xufSlcbmV4cG9ydCBjbGFzcyBRdWVyeUJ1aWxkZXJTZXJ2aWNlIHtcbiAgXG4gIGNvbnN0cnVjdG9yKCkgeyB9XG5cbiAgLyoqXG4gICAqIENvbnZlcnRzIGEgcXVlcnkgZ3JvdXAgdG8gYSBodW1hbi1yZWFkYWJsZSBzdHJpbmdcbiAgICogQHBhcmFtIHF1ZXJ5IFRoZSBxdWVyeSBncm91cCB0byBjb252ZXJ0XG4gICAqIEByZXR1cm5zIEEgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBxdWVyeVxuICAgKi9cbiAgcXVlcnlUb1N0cmluZyhxdWVyeTogUXVlcnlHcm91cCk6IHN0cmluZyB7XG4gICAgaWYgKCFxdWVyeSB8fCAhcXVlcnkucnVsZXMgfHwgcXVlcnkucnVsZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gJyc7XG4gICAgfVxuXG4gICAgcmV0dXJuIHF1ZXJ5LnJ1bGVzXG4gICAgICAubWFwKChydWxlOiBhbnkpID0+IHtcbiAgICAgICAgaWYgKHJ1bGUuY29uZGl0aW9uKSB7XG4gICAgICAgICAgcmV0dXJuIGAoJHt0aGlzLnF1ZXJ5VG9TdHJpbmcocnVsZSBhcyBRdWVyeUdyb3VwKX0pYDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICByZXR1cm4gYCR7cnVsZS5maWVsZH0gJHtydWxlLm9wZXJhdG9yfSAke3J1bGUudmFsdWV9YDtcbiAgICAgICAgfVxuICAgICAgfSlcbiAgICAgIC5qb2luKGAgJHtxdWVyeS5jb25kaXRpb24udG9VcHBlckNhc2UoKX0gYCk7XG4gIH1cblxuICAvKipcbiAgICogQ29udmVydHMgYSBxdWVyeSBncm91cCB0byBhIEpTT04gb2JqZWN0IHRoYXQgY2FuIGJlIHVzZWQgaW4gQVBJIHJlcXVlc3RzXG4gICAqIEBwYXJhbSBxdWVyeSBUaGUgcXVlcnkgZ3JvdXAgdG8gY29udmVydFxuICAgKiBAcmV0dXJucyBBIEpTT04gb2JqZWN0IHJlcHJlc2VudGF0aW9uIG9mIHRoZSBxdWVyeVxuICAgKi9cbiAgcXVlcnlUb0pzb24ocXVlcnk6IFF1ZXJ5R3JvdXApOiBhbnkge1xuICAgIGlmICghcXVlcnkgfHwgIXF1ZXJ5LnJ1bGVzIHx8IHF1ZXJ5LnJ1bGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIHt9O1xuICAgIH1cblxuICAgIHJldHVybiB7XG4gICAgICBjb25kaXRpb246IHF1ZXJ5LmNvbmRpdGlvbixcbiAgICAgIHJ1bGVzOiBxdWVyeS5ydWxlcy5tYXAoKHJ1bGU6IGFueSkgPT4ge1xuICAgICAgICBpZiAocnVsZS5jb25kaXRpb24pIHtcbiAgICAgICAgICByZXR1cm4gdGhpcy5xdWVyeVRvSnNvbihydWxlIGFzIFF1ZXJ5R3JvdXApO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBmaWVsZDogcnVsZS5maWVsZCxcbiAgICAgICAgICAgIG9wZXJhdG9yOiBydWxlLm9wZXJhdG9yLFxuICAgICAgICAgICAgdmFsdWU6IHJ1bGUudmFsdWVcbiAgICAgICAgICB9O1xuICAgICAgICB9XG4gICAgICB9KVxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogVmFsaWRhdGVzIGEgcXVlcnkgZ3JvdXAgdG8gZW5zdXJlIGFsbCBydWxlcyBoYXZlIHZhbGlkIHZhbHVlc1xuICAgKiBAcGFyYW0gcXVlcnkgVGhlIHF1ZXJ5IGdyb3VwIHRvIHZhbGlkYXRlXG4gICAqIEByZXR1cm5zIFRydWUgaWYgdGhlIHF1ZXJ5IGlzIHZhbGlkLCBmYWxzZSBvdGhlcndpc2VcbiAgICovXG4gIGlzUXVlcnlWYWxpZChxdWVyeTogUXVlcnlHcm91cCk6IGJvb2xlYW4ge1xuICAgIGlmICghcXVlcnkgfHwgIXF1ZXJ5LnJ1bGVzIHx8IHF1ZXJ5LnJ1bGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cblxuICAgIHJldHVybiBxdWVyeS5ydWxlcy5ldmVyeSgocnVsZTogYW55KSA9PiB7XG4gICAgICBpZiAocnVsZS5jb25kaXRpb24pIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuaXNRdWVyeVZhbGlkKHJ1bGUgYXMgUXVlcnlHcm91cCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZXR1cm4gcnVsZS5maWVsZCAmJiBydWxlLm9wZXJhdG9yICYmIHJ1bGUudmFsdWUgIT09IHVuZGVmaW5lZCAmJiBydWxlLnZhbHVlICE9PSBudWxsO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgYW4gZW1wdHkgcXVlcnkgZ3JvdXBcbiAgICogQHBhcmFtIGNvbmRpdGlvbiBUaGUgY29uZGl0aW9uIHRvIHVzZSAoYW5kL29yKVxuICAgKiBAcmV0dXJucyBBbiBlbXB0eSBxdWVyeSBncm91cFxuICAgKi9cbiAgY3JlYXRlRW1wdHlRdWVyeShjb25kaXRpb246ICdhbmQnIHwgJ29yJyA9ICdhbmQnKTogUXVlcnlHcm91cCB7XG4gICAgcmV0dXJuIHtcbiAgICAgIGNvbmRpdGlvbixcbiAgICAgIHJ1bGVzOiBbXVxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyBhIHNpbXBsZSBxdWVyeSB3aXRoIGEgc2luZ2xlIHJ1bGVcbiAgICogQHBhcmFtIGZpZWxkIFRoZSBmaWVsZCB0byBmaWx0ZXIgb25cbiAgICogQHBhcmFtIG9wZXJhdG9yIFRoZSBvcGVyYXRvciB0byB1c2VcbiAgICogQHBhcmFtIHZhbHVlIFRoZSB2YWx1ZSB0byBmaWx0ZXIgYnlcbiAgICogQHJldHVybnMgQSBxdWVyeSBncm91cCB3aXRoIGEgc2luZ2xlIHJ1bGVcbiAgICovXG4gIGNyZWF0ZVNpbXBsZVF1ZXJ5KGZpZWxkOiBzdHJpbmcsIG9wZXJhdG9yOiBzdHJpbmcsIHZhbHVlOiBhbnkpOiBRdWVyeUdyb3VwIHtcbiAgICByZXR1cm4ge1xuICAgICAgY29uZGl0aW9uOiAnYW5kJyxcbiAgICAgIHJ1bGVzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBmaWVsZCxcbiAgICAgICAgICBvcGVyYXRvcixcbiAgICAgICAgICB2YWx1ZVxuICAgICAgICB9XG4gICAgICBdXG4gICAgfTtcbiAgfVxufVxuIl19
@@ -2,7 +2,7 @@ import { Component } from '@angular/core';
2
2
  import { FieldArrayType } from '@ngx-formly/core';
3
3
  import { CommonModule } from '@angular/common';
4
4
  import { FormlyModule } from '@ngx-formly/core';
5
- import { FormGroup, FormControl, ReactiveFormsModule } from '@angular/forms';
5
+ import { ReactiveFormsModule } from '@angular/forms';
6
6
  import { get, keys } from 'lodash';
7
7
  import { ButtonComponent } from '../button/button.component';
8
8
  import { IconComponent } from '../icon/icon.component';
@@ -13,12 +13,10 @@ import { MatNativeDateModule, MatOptionModule } from '@angular/material/core';
13
13
  import { MatIconModule } from '@angular/material/icon';
14
14
  import { trigger, style, transition, animate } from '@angular/animations';
15
15
  import { MatDatepickerModule } from '@angular/material/datepicker';
16
+ import { MatTooltipModule } from '@angular/material/tooltip';
16
17
  import * as i0 from "@angular/core";
17
- import * as i1 from "@angular/forms";
18
- import * as i2 from "@ngx-formly/core";
19
- import * as i3 from "@angular/material/form-field";
20
- import * as i4 from "@angular/material/input";
21
- import * as i5 from "@angular/material/datepicker";
18
+ import * as i1 from "@ngx-formly/core";
19
+ import * as i2 from "@angular/material/tooltip";
22
20
  export class QueryBuilderFormlyComponent extends FieldArrayType {
23
21
  ngOnInit() {
24
22
  }
@@ -49,19 +47,6 @@ export class QueryBuilderFormlyComponent extends FieldArrayType {
49
47
  }
50
48
  return field;
51
49
  }
52
- getFilterType(subField) {
53
- const parentValue = get(subField, 'formControl.parent.value');
54
- if (!parentValue)
55
- return undefined;
56
- // Look through all keys of the parent value
57
- for (const key of keys(parentValue)) {
58
- const val = parentValue[key];
59
- if (val && typeof val === 'object' && 'filterType' in val) {
60
- return val.filterType;
61
- }
62
- }
63
- return undefined;
64
- }
65
50
  // reset(): void {
66
51
  // this.formControl.reset();
67
52
  // }
@@ -136,138 +121,144 @@ export class QueryBuilderFormlyComponent extends FieldArrayType {
136
121
  console.log(`🎯 Restored ${restoredCount} date range fields`);
137
122
  }
138
123
  // Override add method to protect existing date range fields
139
- add(i, initialModel) {
140
- console.log('🎯 Add operation started - protecting existing date range fields');
141
- // BACKUP ALL DATE RANGE FIELDS BEFORE add() operation
142
- const allDateRangeBackups = this.backupAllDateRangeFields();
143
- // Call the parent add method
144
- super.add(i, initialModel);
145
- // RESTORE ALL DATE RANGE FIELDS after add() with multiple attempts
146
- setTimeout(() => {
147
- console.log('🎯 First restoration attempt');
148
- this.restoreAllDateRangeFields(allDateRangeBackups);
149
- }, 0);
150
- setTimeout(() => {
151
- console.log('🎯 Second restoration attempt (after handleOperatorChange)');
152
- this.restoreAllDateRangeFields(allDateRangeBackups);
153
- }, 100);
154
- setTimeout(() => {
155
- console.log('🎯 Final restoration attempt');
156
- this.restoreAllDateRangeFields(allDateRangeBackups);
157
- }, 300);
124
+ // override add(i?: number, initialModel?: any): void {
125
+ // console.log('🎯 Add operation started - protecting existing date range fields');
126
+ // // BACKUP ALL DATE RANGE FIELDS BEFORE add() operation
127
+ // const allDateRangeBackups = this.backupAllDateRangeFields();
128
+ // // Call the parent add method
129
+ // super.add(i, initialModel);
130
+ // // RESTORE ALL DATE RANGE FIELDS after add() with multiple attempts
131
+ // setTimeout(() => {
132
+ // console.log('🎯 First restoration attempt');
133
+ // this.restoreAllDateRangeFields(allDateRangeBackups);
134
+ // }, 0);
135
+ // setTimeout(() => {
136
+ // console.log('🎯 Second restoration attempt (after handleOperatorChange)');
137
+ // this.restoreAllDateRangeFields(allDateRangeBackups);
138
+ // }, 100);
139
+ // setTimeout(() => {
140
+ // console.log('🎯 Final restoration attempt');
141
+ // this.restoreAllDateRangeFields(allDateRangeBackups);
142
+ // }, 300);
143
+ // }
144
+ isRowValidForCopy(index) {
145
+ const rowModel = this.formControl.at(index)?.value;
146
+ if (!rowModel || typeof rowModel !== 'object')
147
+ return false;
148
+ // Check if required fields are filled: attribute, operators, and either filterValue or filterValueDateRange
149
+ const hasAttribute = rowModel.attribute && typeof rowModel.attribute === 'object';
150
+ const hasOperator = rowModel.operators && rowModel.operators !== '';
151
+ // Check filterValue (for non-date-range fields) - can be empty string, but must be defined
152
+ const hasFilterValue = 'filterValue' in rowModel &&
153
+ rowModel.filterValue !== undefined &&
154
+ rowModel.filterValue !== null;
155
+ // Check filterValueDateRange (for date range fields with BETWEEN operator)
156
+ const hasDateRangeValue = rowModel.filterValueDateRange &&
157
+ typeof rowModel.filterValueDateRange === 'object' &&
158
+ (rowModel.filterValueDateRange.start || rowModel.filterValueDateRange.end);
159
+ // Row is valid if it has attribute, operator, and at least one value field
160
+ return hasAttribute && hasOperator && (hasFilterValue || hasDateRangeValue);
161
+ }
162
+ reset() {
163
+ this.field.fieldGroup.splice(0, this.field.fieldGroup.length); // clear
164
+ this.add(); // start fresh with one row
165
+ }
166
+ getFilterType(subField) {
167
+ const parentValue = get(subField, 'formControl.parent.value');
168
+ if (!parentValue)
169
+ return undefined;
170
+ // Look through all keys of the parent value
171
+ for (const key of keys(parentValue)) {
172
+ const val = parentValue[key];
173
+ if (val && typeof val === 'object' && 'filterType' in val) {
174
+ return get(val, 'filterType');
175
+ }
176
+ }
177
+ return undefined;
178
+ }
179
+ /**
180
+ * Helper method to restore a simple field value
181
+ */
182
+ restoreSimpleField(formGroup, fieldName, value, backupKey) {
183
+ const ctrl = formGroup?.get(fieldName);
184
+ if (ctrl && value !== undefined) {
185
+ if (backupKey && !formGroup.value[backupKey]) {
186
+ formGroup.patchValue({ [backupKey]: value });
187
+ }
188
+ ctrl.setValue(value);
189
+ }
190
+ }
191
+ /**
192
+ * Helper method to restore a fieldGroup with nested controls (like date range)
193
+ */
194
+ restoreFieldGroup(formGroup, fieldName, value) {
195
+ const ctrl = formGroup?.get(fieldName);
196
+ if (ctrl && value && typeof value === 'object') {
197
+ Object.keys(value).forEach(key => {
198
+ if (value[key] !== undefined) {
199
+ ctrl.get(key)?.setValue(value[key]);
200
+ }
201
+ });
202
+ }
158
203
  }
204
+ /**
205
+ * Duplicates the row at the given index.
206
+ * A deep–clone of that row's model is inserted right after the original row.
207
+ */
159
208
  copy(index) {
160
209
  const rowModel = this.formControl.at(index)?.value;
161
210
  if (!rowModel)
162
211
  return;
163
212
  const clonedModel = JSON.parse(JSON.stringify(rowModel));
164
- // Carry the value over to the new row so the onInit hook can restore it.
165
- if (clonedModel.filterValue !== undefined) {
166
- clonedModel._copiedValue = clonedModel.filterValue;
167
- }
213
+ // Define fields to copy with their backup keys
214
+ const fieldsToCopy = [
215
+ { field: 'filterValue', backupKey: '_copiedValue' },
216
+ { field: 'filterValueDateRange', backupKey: '_copiedValueDateRange' }
217
+ ];
218
+ // Store copied values for restoration
219
+ fieldsToCopy.forEach(({ field, backupKey }) => {
220
+ if (clonedModel[field] !== undefined) {
221
+ clonedModel[backupKey] = clonedModel[field];
222
+ }
223
+ });
168
224
  // If copying the first row (index 0), the new row (index 1) should have a logical operator
169
- // Set default logical operator to 'and' for the new row
170
225
  if (index === 0) {
171
226
  clonedModel.logicalOperator = 'and';
172
227
  }
173
- // Determine if this is a date range before any modifications
174
- const copiedValue = clonedModel._copiedValue;
175
- const isDateRange = clonedModel.operators === 'BETWEEN' &&
176
- clonedModel.attribute.filterType === 'DATE' &&
177
- copiedValue && typeof copiedValue === 'object' &&
178
- copiedValue.startDate && copiedValue.endDate;
179
- // BACKUP ALL DATE RANGE FIELDS BEFORE add() operation
180
- const allDateRangeBackups = this.backupAllDateRangeFields();
181
228
  this.add(index + 1, clonedModel);
182
- // IMMEDIATELY RESTORE ALL DATE RANGE FIELDS after add()
183
- this.restoreAllDateRangeFields(allDateRangeBackups);
184
- // Force rebuild of the filterValue field by triggering attribute change
229
+ // Force rebuild of fields by triggering attribute change
185
230
  setTimeout(() => {
186
231
  const newRowForm = this.formControl.at(index + 1);
187
232
  const attrCtrl = newRowForm?.get('attribute');
188
- const operatorCtrl = newRowForm?.get('operators');
189
- const filterValueCtrl = newRowForm?.get('filterValue');
190
- if (attrCtrl && operatorCtrl && filterValueCtrl && clonedModel.attribute) {
191
- console.log('🎯 Copy: isDateRange =', isDateRange, 'copiedValue =', copiedValue);
192
- // For date range, set up everything without triggering change events
193
- if (isDateRange) {
194
- console.log('🎯 Copy: Setting up date range without events');
195
- // Set attribute and operator without events
196
- attrCtrl.setValue(clonedModel.attribute, { emitEvent: false });
197
- operatorCtrl.setValue(clonedModel.operators, { emitEvent: false });
198
- // Create FormGroup directly
199
- const dateRangeGroup = new FormGroup({
200
- startDate: new FormControl(copiedValue.startDate),
201
- endDate: new FormControl(copiedValue.endDate)
202
- });
203
- // Replace the FormControl with FormGroup
204
- newRowForm.setControl('filterValue', dateRangeGroup);
205
- // Find the filterValue field in the new row and update its structure
206
- const newRowFieldGroup = this.field.fieldGroup[index + 1];
207
- const filterValueField = newRowFieldGroup?.fieldGroup?.find((f) => f.key === 'filterValue');
208
- if (filterValueField) {
209
- // Update field structure to show fieldGroup (two date pickers)
210
- filterValueField.props = {
211
- ...filterValueField.props,
212
- showDateRange: true,
213
- attribute: clonedModel.attribute,
214
- hasDynamicType: true
215
- };
216
- // Create fieldGroup structure
217
- filterValueField.fieldGroup = [
218
- {
219
- key: 'startDate',
220
- type: 'datepicker',
221
- className: 'query-builder-form-wrapper flex-2',
222
- formControl: dateRangeGroup.get('startDate'),
223
- props: {
224
- placeholder: 'Start date',
225
- label: 'Start Date'
226
- }
227
- },
228
- {
229
- key: 'endDate',
230
- type: 'datepicker',
231
- className: 'query-builder-form-wrapper flex-2',
232
- formControl: dateRangeGroup.get('endDate'),
233
- props: {
234
- placeholder: 'End date',
235
- label: 'End Date'
236
- }
237
- }
238
- ];
239
- // Clear the type since we're using fieldGroup
240
- filterValueField.type = null;
241
- // Update the formControl reference (cast to any to bypass readonly)
242
- filterValueField.formControl = dateRangeGroup;
243
- console.log('🎯 Copy: Date range setup complete', dateRangeGroup.value);
244
- }
245
- }
246
- else {
247
- // For single values, use normal approach
248
- attrCtrl.setValue(null, { emitEvent: false });
249
- operatorCtrl.setValue(null, { emitEvent: false });
250
- setTimeout(() => {
251
- attrCtrl.setValue(clonedModel.attribute, { emitEvent: true });
252
- setTimeout(() => {
253
- operatorCtrl.setValue(clonedModel.operators, { emitEvent: true });
254
- setTimeout(() => {
255
- if (copiedValue !== undefined) {
256
- filterValueCtrl.setValue(copiedValue);
257
- }
258
- }, 50);
259
- }, 50);
260
- }, 10);
261
- }
262
- }
233
+ if (!attrCtrl || !clonedModel.attribute)
234
+ return;
235
+ // Store all values to restore
236
+ const valuesToRestore = {
237
+ operators: clonedModel.operators,
238
+ filterValue: clonedModel._copiedValue,
239
+ filterValueDateRange: clonedModel._copiedValueDateRange
240
+ };
241
+ // Temporarily clear the attribute to force a change event
242
+ attrCtrl.setValue(null, { emitEvent: false });
243
+ // Restore attribute to trigger field type rebuild
244
+ setTimeout(() => {
245
+ attrCtrl.setValue(clonedModel.attribute, { emitEvent: true });
246
+ // Restore operator after attribute is set
247
+ setTimeout(() => {
248
+ this.restoreSimpleField(newRowForm, 'operators', valuesToRestore.operators);
249
+ }, 30);
250
+ // Restore all value fields after rebuild
251
+ setTimeout(() => {
252
+ // Restore simple filterValue
253
+ this.restoreSimpleField(newRowForm, 'filterValue', valuesToRestore.filterValue, '_copiedValue');
254
+ // Restore fieldGroup filterValueDateRange
255
+ this.restoreFieldGroup(newRowForm, 'filterValueDateRange', valuesToRestore.filterValueDateRange);
256
+ }, 100);
257
+ }, 10);
263
258
  });
264
259
  }
265
- reset() {
266
- this.field.fieldGroup.splice(0, this.field.fieldGroup.length); // clear
267
- this.add(); // start fresh with one row
268
- }
269
260
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.4", ngImport: i0, type: QueryBuilderFormlyComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
270
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.2.4", type: QueryBuilderFormlyComponent, isStandalone: true, selector: "sa-query-builder-formly", providers: [], usesInheritance: true, ngImport: i0, template: "<div class=\"sa-query-builder-container\">\r\n <div class=\"query-builder-body\">\r\n @for(subField of field.fieldGroup; track subField.id || $index) {\r\n <div class=\"row align-items-baseline query-row\" [@rowAnimation]>\r\n \r\n @if ($index === 0) {\r\n <span class=\"where-text flex-1\">Where</span>\r\n }\r\n @for (innerField of subField.fieldGroup; track innerField.key || $index) {\r\n \r\n @if(!!innerField.props['hasDynamicType']){\r\n <!-- This is a single field -->\r\n @switch (getFilterType(innerField)) {\r\n \r\n @case ('STRING') {\r\n <formly-field [field]=\"innerField\"></formly-field>\r\n }\r\n @case ('NUMBER') {\r\n <formly-field [field]=\"innerField\"></formly-field>\r\n }\r\n @case ('BOOLEAN') {\r\n <formly-field [field]=\"innerField\"></formly-field>\r\n }\r\n @case ('CURRENCY') {\r\n <formly-field [field]=\"innerField\"></formly-field>\r\n }\r\n @case ('DATE') {\r\n {{innerField.props?.['showDateRange'] || false}}\r\n @if(innerField.props?.['showDateRange']){\r\n <!-- Date Range Mode - Two Date Pickers -->\r\n @for(dateField of innerField.fieldGroup; track dateField.key) {\r\n <mat-form-field [appearance]=\"'outline'\" class=\"query-builder-form-wrapper flex-2\">\r\n <mat-label>{{dateField.key === 'startDate' ? 'Start Date' : 'End Date'}}</mat-label>\r\n <input matInput [matDatepicker]=\"dateField.key === 'startDate' ? startPicker : endPicker\"\r\n [placeholder]=\"dateField.key === 'startDate' ? 'Select start date' : 'Select end date'\"\r\n [formControl]=\"$any(dateField.formControl)\" \r\n [formlyAttributes]=\"dateField\" />\r\n <mat-datepicker-toggle matSuffix\r\n [for]=\"dateField.key === 'startDate' ? startPicker : endPicker\"></mat-datepicker-toggle>\r\n </mat-form-field>\r\n }\r\n <mat-datepicker #startPicker></mat-datepicker>\r\n <mat-datepicker #endPicker></mat-datepicker>\r\n }\r\n @else{\r\n <mat-form-field [appearance]=\"innerField.props?.['appearance'] || 'outline'\"\r\n class=\"query-builder-form-wrapper flex-2\">\r\n <mat-label>{{innerField.key === 'endDate' ? 'End Date' : (innerField.props?.placeholder || 'Select\r\n date')}}</mat-label>\r\n <input matInput [matDatepicker]=\"picker\" [placeholder]=\"innerField.props?.placeholder || 'Select date'\"\r\n [formControl]=\"$any(innerField.formControl)\" [formlyAttributes]=\"innerField\"\r\n [required]=\"!!innerField.props?.required\" />\r\n <mat-datepicker-toggle matSuffix [for]=\"picker\"></mat-datepicker-toggle>\r\n <mat-datepicker #picker></mat-datepicker>\r\n </mat-form-field>\r\n }\r\n }\r\n @case ('SELECT') {\r\n <formly-field [field]=\"createField(innerField)\"></formly-field>\r\n }\r\n @default {\r\n <formly-field [field]=\"createField(innerField)\"></formly-field>\r\n }\r\n\r\n }\r\n }@else{\r\n <formly-field class=\"col\" [field]=\"innerField\">\r\n </formly-field>\r\n }\r\n }\r\n @if(props?.['showCopyButton']){\r\n <sa-icon icon=\"copyOutlined\" class=\"copy-icon pointer\" (click)=\"copy($index)\"></sa-icon>\r\n }\r\n <sa-icon icon=\"deleteIcon\" class=\"delete-icon pointer\" (click)=\"remove($index)\"></sa-icon>\r\n </div>\r\n }\r\n\r\n <!-- Buttons -->\r\n <div class=\"rule-button-container\">\r\n <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\r\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\" type=\"outline\"\r\n size=\"medium\">\r\n </sa-button>\r\n <span class=\"filters-count\">({{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters)</span>\r\n @if(props?.['showResetButton']){\r\n <sa-button iconPosition=\"left\" text=\"Reset\" (onClickEvent)=\"reset()\" type=\"primary\" size=\"medium\">\r\n </sa-button>\r\n }\r\n </div>\r\n </div>\r\n</div>", styles: [".sa-query-builder-container{display:flex}.sa-query-builder-container .query-builder-body{display:flex;flex-direction:column;gap:var(--small-12px);width:-webkit-fill-available}.query-builder-body .row{display:flex;height:2.25rem;justify-content:center;align-items:center;flex:1 0 0;gap:var(--small-8px);width:-webkit-fill-available}.query-row{transition:all .2s ease-in-out;border-radius:var(--small-4px);padding:var(--small-4px);margin:2px 0}.query-row ::ng-deep .query-builder-form-wrapper{transition:all .15s ease-in-out}.query-builder-form-wrapper{width:-webkit-fill-available}::ng-deep .query-builder-form-wrapper .sa-input-field.idle{border-radius:var(--small-4px);border:1px solid var(--grey-100, #EAECF0);background:var(--white, #FFF);height:2.438rem}::ng-deep .query-builder-form-wrapper .sa-input-container{height:3.438rem}::ng-deep .query-builder-form-wrapper .sa-input-container .disabled .sa-input-field input{color:var(--text-low-emphasis, #9B98A3);font-family:var(--font-family, Roboto);font-size:13px;font-style:normal;font-weight:400;line-height:var(--medium-20px);letter-spacing:.25px}.where-text{color:var(--text-medium-emphasis, #6D6979);font-family:var(--font-family-roboto);font-size:var(--small-14px);font-style:normal;font-weight:var(--font-weight-500);line-height:var(--medium-20px);letter-spacing:.1px;width:80px}.flex-1{flex:1 1 0;min-width:80px}.flex-2{flex:2 1 0;min-width:120px}.flex-4{flex:4 1 0;min-width:200px}.date-range-container{display:flex;gap:var(--small-8px);align-items:center;width:100%}.date-range-container .value-input{flex:1;min-width:120px}.copy-icon,.delete-icon{transition:all .2s ease-in-out;padding:4px;border-radius:4px;opacity:.7}.copy-icon:hover,.delete-icon:hover{opacity:1;transform:scale(1.1);background-color:#0000000d}.copy-icon:active,.delete-icon:active{transform:scale(.95)}.copy-icon:hover{color:var(--primary-500, #7C3AED)}.delete-icon:hover{color:var(--error-500, #EF4444)}.rule-button-container{transition:all .2s ease-in-out}.rule-button-container sa-button{transition:all .2s ease-in-out}.rule-button-container sa-button:hover{transform:translateY(-1px)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: FormlyModule }, { kind: "component", type: i2.FormlyField, selector: "formly-field", inputs: ["field"] }, { kind: "directive", type: i2.ɵFormlyAttributes, selector: "[formlyAttributes]", inputs: ["formlyAttributes", "id"] }, { kind: "component", type: ButtonComponent, selector: "sa-button", inputs: ["id", "type", "state", "size", "text", "ImagePath", "icon", "iconPosition", "href", "hrefTarget", "width", "isSubmit", "buttonIconSize", "showSpinner"], outputs: ["onClickEvent", "onMouseInEvent", "onMouseOutEvent"] }, { kind: "component", type: IconComponent, selector: "sa-icon", inputs: ["img", "imgWidth", "imgHeight", "icon", "size", "color", "iconPath", "iconUrl", "customClass", "href", "hrefTarget", "iconPosition"], outputs: ["onClickEvent"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3.MatLabel, selector: "mat-label" }, { kind: "directive", type: i3.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatIconModule }, { kind: "ngmodule", type: MatDatepickerModule }, { kind: "component", type: i5.MatDatepicker, selector: "mat-datepicker", exportAs: ["matDatepicker"] }, { kind: "directive", type: i5.MatDatepickerInput, selector: "input[matDatepicker]", inputs: ["matDatepicker", "min", "max", "matDatepickerFilter"], exportAs: ["matDatepickerInput"] }, { kind: "component", type: i5.MatDatepickerToggle, selector: "mat-datepicker-toggle", inputs: ["for", "tabIndex", "aria-label", "disabled", "disableRipple"], exportAs: ["matDatepickerToggle"] }, { kind: "ngmodule", type: MatNativeDateModule }], animations: [
261
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.2.4", type: QueryBuilderFormlyComponent, isStandalone: true, selector: "sa-query-builder-formly", providers: [], usesInheritance: true, ngImport: i0, template: "<div class=\"sa-query-builder-container\">\n <div class=\"query-builder-body\">\n @for(row of field.fieldGroup; track row.id || $index; let last = $last) {\n\n <div class=\"row align-items-baseline query-row\">\n @if ($index === 0) {\n <span class=\"where-text flex-1\">Where the:</span>\n }\n @for (ruleField of row.fieldGroup; track ruleField.key || $index) {\n @if(!!ruleField?.props?.['hasDynamicType']){\n @switch (getFilterType(ruleField)) {\n @case ('STRING') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('NUMBER') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('BOOLEAN') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('CURRENCY') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('DATE') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('SELECT') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @default {\n <formly-field [field]=\"createField(ruleField)\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n }\n }@else{\n <formly-field class=\"col\" [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n }\n <div class=\"rule-icon-container\">\n @if(props?.['showCopyButton']){\n <sa-icon icon=\"copyOutlined\" size=\"20\" matTooltip=\"Copy\" class=\"copy-icon pointer {{(field?.fieldGroup?.length === (props?.['maxRows'] || 5)) || !isRowValidForCopy($index) ? 'disabled-part' : ''}}\" (click)=\"isRowValidForCopy($index) && copy($index)\"></sa-icon>\n }\n @if(props?.['showDeleteRowButton']){\n <sa-icon icon=\"deleteIcon\" size=\"20\" matTooltip=\"Delete\" class=\"delete-icon pointer {{field?.fieldGroup?.length === (1) ? 'disabled-part' : ''}}\" (click)=\"remove($index)\"></sa-icon>\n }\n </div>\n </div>\n @if(!last || (last && props?.['showLastDivider'])){\n <div class=\"segment-divider\"></div>\n }\n }\n\n <!-- Buttons -->\n <div class=\"rule-button-container\">\n @if(props?.['showAddFilterButton']){\n <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n type=\"outline\" size=\"medium\">\n </sa-button>\n }\n @if(props?.['showCount']){\n <span class=\"filters-count\">{{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters</span>\n }\n @if(props?.['showResetButton']){\n <sa-button iconPosition=\"left\" text=\"Reset\" (onClickEvent)=\"reset()\" type=\"primary\" size=\"medium\">\n </sa-button>\n }\n </div>\n </div>\n</div>", styles: [".sa-query-builder-container{display:flex}.sa-query-builder-container .query-builder-body{display:flex;flex-direction:column;gap:var(--small-12px);width:-webkit-fill-available}.query-builder-body .row{display:flex;height:2.25rem;justify-content:center;align-items:center;flex:1 0 0;gap:var(--small-8px);width:-webkit-fill-available}.query-row{transition:all .2s ease-in-out;border-radius:var(--small-4px);padding:var(--small-4px);margin:2px 0}.query-row ::ng-deep .query-builder-form-wrapper{transition:all .15s ease-in-out}.query-builder-form-wrapper{width:-webkit-fill-available}::ng-deep .query-builder-form-wrapper .sa-input-field.idle{border-radius:var(--small-4px);border:1px solid var(--grey-100, #EAECF0);background:var(--white, #FFF);height:2.438rem}::ng-deep .query-builder-form-wrapper .sa-input-container{height:3.438rem}::ng-deep .query-builder-form-wrapper .sa-input-container .disabled .sa-input-field input{color:var(--text-low-emphasis, #9B98A3);font-family:var(--font-family, Roboto);font-size:13px;font-style:normal;font-weight:400;line-height:var(--medium-20px);letter-spacing:.25px}.where-text{color:var(--text-medium-emphasis, #6D6979);font-family:var(--font-family-roboto);font-size:var(--small-14px);font-style:normal;font-weight:var(--font-weight-500);line-height:var(--medium-20px);letter-spacing:.1px;width:80px}.flex-1{flex:1 1 0;min-width:80px}.flex-2{flex:2 1 0;min-width:120px}.flex-4{flex:4 1 0;min-width:200px}.date-range-container{display:flex;gap:var(--small-8px);align-items:center;width:100%}.date-range-container .value-input{flex:1;min-width:120px}.copy-icon,.delete-icon{transition:all .2s ease-in-out;padding:4px;border-radius:4px;opacity:.7}.copy-icon:hover,.delete-icon:hover{opacity:1;transform:scale(1.1);background-color:#0000000d}.copy-icon:active,.delete-icon:active{transform:scale(.95)}.copy-icon:hover{color:var(--primary-500, #7C3AED)}.delete-icon:hover{color:var(--error-500, #EF4444)}.rule-button-container{transition:all .2s ease-in-out}.rule-button-container sa-button{transition:all .2s ease-in-out}.rule-button-container sa-button:hover{transform:translateY(-1px)}::ng-deep .query-builder-form-wrapper .filtervalue-class formly-group{display:flex;gap:var(--small-8px)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "ngmodule", type: FormlyModule }, { kind: "component", type: i1.FormlyField, selector: "formly-field", inputs: ["field"] }, { kind: "component", type: ButtonComponent, selector: "sa-button", inputs: ["id", "type", "state", "size", "text", "ImagePath", "icon", "iconPosition", "href", "hrefTarget", "width", "isSubmit", "buttonIconSize", "showSpinner"], outputs: ["onClickEvent", "onMouseInEvent", "onMouseOutEvent"] }, { kind: "component", type: IconComponent, selector: "sa-icon", inputs: ["img", "imgWidth", "imgHeight", "icon", "size", "color", "iconPath", "iconUrl", "customClass", "href", "hrefTarget", "iconPosition"], outputs: ["onClickEvent"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatInputModule }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatIconModule }, { kind: "ngmodule", type: MatDatepickerModule }, { kind: "ngmodule", type: MatNativeDateModule }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i2.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }], animations: [
271
262
  trigger('rowAnimation', [
272
263
  transition(':enter', [
273
264
  style({
@@ -342,6 +333,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImpor
342
333
  MatIconModule,
343
334
  MatDatepickerModule,
344
335
  MatNativeDateModule,
345
- ], template: "<div class=\"sa-query-builder-container\">\r\n <div class=\"query-builder-body\">\r\n @for(subField of field.fieldGroup; track subField.id || $index) {\r\n <div class=\"row align-items-baseline query-row\" [@rowAnimation]>\r\n \r\n @if ($index === 0) {\r\n <span class=\"where-text flex-1\">Where</span>\r\n }\r\n @for (innerField of subField.fieldGroup; track innerField.key || $index) {\r\n \r\n @if(!!innerField.props['hasDynamicType']){\r\n <!-- This is a single field -->\r\n @switch (getFilterType(innerField)) {\r\n \r\n @case ('STRING') {\r\n <formly-field [field]=\"innerField\"></formly-field>\r\n }\r\n @case ('NUMBER') {\r\n <formly-field [field]=\"innerField\"></formly-field>\r\n }\r\n @case ('BOOLEAN') {\r\n <formly-field [field]=\"innerField\"></formly-field>\r\n }\r\n @case ('CURRENCY') {\r\n <formly-field [field]=\"innerField\"></formly-field>\r\n }\r\n @case ('DATE') {\r\n {{innerField.props?.['showDateRange'] || false}}\r\n @if(innerField.props?.['showDateRange']){\r\n <!-- Date Range Mode - Two Date Pickers -->\r\n @for(dateField of innerField.fieldGroup; track dateField.key) {\r\n <mat-form-field [appearance]=\"'outline'\" class=\"query-builder-form-wrapper flex-2\">\r\n <mat-label>{{dateField.key === 'startDate' ? 'Start Date' : 'End Date'}}</mat-label>\r\n <input matInput [matDatepicker]=\"dateField.key === 'startDate' ? startPicker : endPicker\"\r\n [placeholder]=\"dateField.key === 'startDate' ? 'Select start date' : 'Select end date'\"\r\n [formControl]=\"$any(dateField.formControl)\" \r\n [formlyAttributes]=\"dateField\" />\r\n <mat-datepicker-toggle matSuffix\r\n [for]=\"dateField.key === 'startDate' ? startPicker : endPicker\"></mat-datepicker-toggle>\r\n </mat-form-field>\r\n }\r\n <mat-datepicker #startPicker></mat-datepicker>\r\n <mat-datepicker #endPicker></mat-datepicker>\r\n }\r\n @else{\r\n <mat-form-field [appearance]=\"innerField.props?.['appearance'] || 'outline'\"\r\n class=\"query-builder-form-wrapper flex-2\">\r\n <mat-label>{{innerField.key === 'endDate' ? 'End Date' : (innerField.props?.placeholder || 'Select\r\n date')}}</mat-label>\r\n <input matInput [matDatepicker]=\"picker\" [placeholder]=\"innerField.props?.placeholder || 'Select date'\"\r\n [formControl]=\"$any(innerField.formControl)\" [formlyAttributes]=\"innerField\"\r\n [required]=\"!!innerField.props?.required\" />\r\n <mat-datepicker-toggle matSuffix [for]=\"picker\"></mat-datepicker-toggle>\r\n <mat-datepicker #picker></mat-datepicker>\r\n </mat-form-field>\r\n }\r\n }\r\n @case ('SELECT') {\r\n <formly-field [field]=\"createField(innerField)\"></formly-field>\r\n }\r\n @default {\r\n <formly-field [field]=\"createField(innerField)\"></formly-field>\r\n }\r\n\r\n }\r\n }@else{\r\n <formly-field class=\"col\" [field]=\"innerField\">\r\n </formly-field>\r\n }\r\n }\r\n @if(props?.['showCopyButton']){\r\n <sa-icon icon=\"copyOutlined\" class=\"copy-icon pointer\" (click)=\"copy($index)\"></sa-icon>\r\n }\r\n <sa-icon icon=\"deleteIcon\" class=\"delete-icon pointer\" (click)=\"remove($index)\"></sa-icon>\r\n </div>\r\n }\r\n\r\n <!-- Buttons -->\r\n <div class=\"rule-button-container\">\r\n <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\r\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\" type=\"outline\"\r\n size=\"medium\">\r\n </sa-button>\r\n <span class=\"filters-count\">({{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters)</span>\r\n @if(props?.['showResetButton']){\r\n <sa-button iconPosition=\"left\" text=\"Reset\" (onClickEvent)=\"reset()\" type=\"primary\" size=\"medium\">\r\n </sa-button>\r\n }\r\n </div>\r\n </div>\r\n</div>", styles: [".sa-query-builder-container{display:flex}.sa-query-builder-container .query-builder-body{display:flex;flex-direction:column;gap:var(--small-12px);width:-webkit-fill-available}.query-builder-body .row{display:flex;height:2.25rem;justify-content:center;align-items:center;flex:1 0 0;gap:var(--small-8px);width:-webkit-fill-available}.query-row{transition:all .2s ease-in-out;border-radius:var(--small-4px);padding:var(--small-4px);margin:2px 0}.query-row ::ng-deep .query-builder-form-wrapper{transition:all .15s ease-in-out}.query-builder-form-wrapper{width:-webkit-fill-available}::ng-deep .query-builder-form-wrapper .sa-input-field.idle{border-radius:var(--small-4px);border:1px solid var(--grey-100, #EAECF0);background:var(--white, #FFF);height:2.438rem}::ng-deep .query-builder-form-wrapper .sa-input-container{height:3.438rem}::ng-deep .query-builder-form-wrapper .sa-input-container .disabled .sa-input-field input{color:var(--text-low-emphasis, #9B98A3);font-family:var(--font-family, Roboto);font-size:13px;font-style:normal;font-weight:400;line-height:var(--medium-20px);letter-spacing:.25px}.where-text{color:var(--text-medium-emphasis, #6D6979);font-family:var(--font-family-roboto);font-size:var(--small-14px);font-style:normal;font-weight:var(--font-weight-500);line-height:var(--medium-20px);letter-spacing:.1px;width:80px}.flex-1{flex:1 1 0;min-width:80px}.flex-2{flex:2 1 0;min-width:120px}.flex-4{flex:4 1 0;min-width:200px}.date-range-container{display:flex;gap:var(--small-8px);align-items:center;width:100%}.date-range-container .value-input{flex:1;min-width:120px}.copy-icon,.delete-icon{transition:all .2s ease-in-out;padding:4px;border-radius:4px;opacity:.7}.copy-icon:hover,.delete-icon:hover{opacity:1;transform:scale(1.1);background-color:#0000000d}.copy-icon:active,.delete-icon:active{transform:scale(.95)}.copy-icon:hover{color:var(--primary-500, #7C3AED)}.delete-icon:hover{color:var(--error-500, #EF4444)}.rule-button-container{transition:all .2s ease-in-out}.rule-button-container sa-button{transition:all .2s ease-in-out}.rule-button-container sa-button:hover{transform:translateY(-1px)}\n"] }]
336
+ MatTooltipModule
337
+ ], template: "<div class=\"sa-query-builder-container\">\n <div class=\"query-builder-body\">\n @for(row of field.fieldGroup; track row.id || $index; let last = $last) {\n\n <div class=\"row align-items-baseline query-row\">\n @if ($index === 0) {\n <span class=\"where-text flex-1\">Where the:</span>\n }\n @for (ruleField of row.fieldGroup; track ruleField.key || $index) {\n @if(!!ruleField?.props?.['hasDynamicType']){\n @switch (getFilterType(ruleField)) {\n @case ('STRING') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('NUMBER') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('BOOLEAN') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('CURRENCY') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('DATE') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @case ('SELECT') {\n <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n @default {\n <formly-field [field]=\"createField(ruleField)\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n }\n }@else{\n <formly-field class=\"col\" [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n }\n }\n <div class=\"rule-icon-container\">\n @if(props?.['showCopyButton']){\n <sa-icon icon=\"copyOutlined\" size=\"20\" matTooltip=\"Copy\" class=\"copy-icon pointer {{(field?.fieldGroup?.length === (props?.['maxRows'] || 5)) || !isRowValidForCopy($index) ? 'disabled-part' : ''}}\" (click)=\"isRowValidForCopy($index) && copy($index)\"></sa-icon>\n }\n @if(props?.['showDeleteRowButton']){\n <sa-icon icon=\"deleteIcon\" size=\"20\" matTooltip=\"Delete\" class=\"delete-icon pointer {{field?.fieldGroup?.length === (1) ? 'disabled-part' : ''}}\" (click)=\"remove($index)\"></sa-icon>\n }\n </div>\n </div>\n @if(!last || (last && props?.['showLastDivider'])){\n <div class=\"segment-divider\"></div>\n }\n }\n\n <!-- Buttons -->\n <div class=\"rule-button-container\">\n @if(props?.['showAddFilterButton']){\n <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n type=\"outline\" size=\"medium\">\n </sa-button>\n }\n @if(props?.['showCount']){\n <span class=\"filters-count\">{{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters</span>\n }\n @if(props?.['showResetButton']){\n <sa-button iconPosition=\"left\" text=\"Reset\" (onClickEvent)=\"reset()\" type=\"primary\" size=\"medium\">\n </sa-button>\n }\n </div>\n </div>\n</div>", styles: [".sa-query-builder-container{display:flex}.sa-query-builder-container .query-builder-body{display:flex;flex-direction:column;gap:var(--small-12px);width:-webkit-fill-available}.query-builder-body .row{display:flex;height:2.25rem;justify-content:center;align-items:center;flex:1 0 0;gap:var(--small-8px);width:-webkit-fill-available}.query-row{transition:all .2s ease-in-out;border-radius:var(--small-4px);padding:var(--small-4px);margin:2px 0}.query-row ::ng-deep .query-builder-form-wrapper{transition:all .15s ease-in-out}.query-builder-form-wrapper{width:-webkit-fill-available}::ng-deep .query-builder-form-wrapper .sa-input-field.idle{border-radius:var(--small-4px);border:1px solid var(--grey-100, #EAECF0);background:var(--white, #FFF);height:2.438rem}::ng-deep .query-builder-form-wrapper .sa-input-container{height:3.438rem}::ng-deep .query-builder-form-wrapper .sa-input-container .disabled .sa-input-field input{color:var(--text-low-emphasis, #9B98A3);font-family:var(--font-family, Roboto);font-size:13px;font-style:normal;font-weight:400;line-height:var(--medium-20px);letter-spacing:.25px}.where-text{color:var(--text-medium-emphasis, #6D6979);font-family:var(--font-family-roboto);font-size:var(--small-14px);font-style:normal;font-weight:var(--font-weight-500);line-height:var(--medium-20px);letter-spacing:.1px;width:80px}.flex-1{flex:1 1 0;min-width:80px}.flex-2{flex:2 1 0;min-width:120px}.flex-4{flex:4 1 0;min-width:200px}.date-range-container{display:flex;gap:var(--small-8px);align-items:center;width:100%}.date-range-container .value-input{flex:1;min-width:120px}.copy-icon,.delete-icon{transition:all .2s ease-in-out;padding:4px;border-radius:4px;opacity:.7}.copy-icon:hover,.delete-icon:hover{opacity:1;transform:scale(1.1);background-color:#0000000d}.copy-icon:active,.delete-icon:active{transform:scale(.95)}.copy-icon:hover{color:var(--primary-500, #7C3AED)}.delete-icon:hover{color:var(--error-500, #EF4444)}.rule-button-container{transition:all .2s ease-in-out}.rule-button-container sa-button{transition:all .2s ease-in-out}.rule-button-container sa-button:hover{transform:translateY(-1px)}::ng-deep .query-builder-form-wrapper .filtervalue-class formly-group{display:flex;gap:var(--small-8px)}\n"] }]
346
338
  }] });
347
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-builder-formly.component.js","sourceRoot":"","sources":["../../../../../projects/component-library/src/lib/query-builder-formly/query-builder-formly.component.ts","../../../../../projects/component-library/src/lib/query-builder-formly/query-builder-formly.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit } from '@angular/core';\r\nimport { FieldArrayType, FormlyFieldConfig } from '@ngx-formly/core';\r\nimport { CommonModule } from '@angular/common'; \r\nimport { FormlyModule } from '@ngx-formly/core';\r\nimport { FormGroup, FormControl, ReactiveFormsModule } from '@angular/forms';\r\nimport { get, keys } from 'lodash';\r\nimport { ButtonComponent } from '../button/button.component';\r\nimport { IconComponent } from '../icon/icon.component';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatNativeDateModule, MatOptionModule } from '@angular/material/core';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { trigger, state, style, transition, animate, query, stagger } from '@angular/animations';\r\nimport { MatDatepickerModule } from '@angular/material/datepicker';\r\n\r\n\r\n@Component({\r\n  selector: 'sa-query-builder-formly',\r\n  templateUrl: './query-builder-formly.component.html',\r\n  styleUrls: ['./query-builder-formly.component.scss'],\r\n  standalone: true,\r\n  providers: [],\r\n  animations: [\r\n    trigger('rowAnimation', [\r\n      transition(':enter', [\r\n        style({ \r\n          opacity: 0, \r\n          transform: 'translateY(-10px) scale(0.95)',\r\n          height: '0px',\r\n          overflow: 'hidden'\r\n        }),\r\n        animate('300ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({ \r\n          opacity: 1, \r\n          transform: 'translateY(0) scale(1)',\r\n          height: '*'\r\n        }))\r\n      ]),\r\n      transition(':leave', [\r\n        style({ \r\n          opacity: 1, \r\n          transform: 'translateY(0) scale(1)',\r\n          height: '*'\r\n        }),\r\n        animate('250ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({ \r\n          opacity: 0, \r\n          transform: 'translateY(-10px) scale(0.95)',\r\n          height: '0px',\r\n          overflow: 'hidden'\r\n        }))\r\n      ])\r\n    ])\r\n  ],\r\n  imports: [\r\n    CommonModule,\r\n    ReactiveFormsModule,\r\n    FormlyModule,\r\n    ButtonComponent,\r\n    IconComponent,\r\n    MatFormFieldModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    MatOptionModule,\r\n    MatIconModule,\r\n    MatDatepickerModule,\r\n    MatNativeDateModule,\r\n  ]\r\n})\r\nexport class QueryBuilderFormlyComponent extends FieldArrayType implements OnInit {\r\n  \r\n  ngOnInit(): void {\r\n    \r\n    \r\n  }\r\n  \r\n  getRowCondition(index: number): 'and' | 'or' {\r\n      if (index === 0) return 'and'; \r\n      \r\n      const ruleGroup = this.formControl.at(index) as FormGroup;\r\n      return ruleGroup.get('logicalOperator')?.value || 'and';\r\n    }\r\n\r\n    /**\r\n   * Sets the condition for a specific row\r\n   * @param index The index of the row\r\n   * @param condition The condition to set (and/or)\r\n   */\r\n  setRowCondition(index: number, condition: 'and' | 'or'): void {\r\n    if (index === 0) return; // First row doesn't have a logical operator\r\n    \r\n    const ruleGroup = this.formControl.at(index) as FormGroup;\r\n    ruleGroup.get('logicalOperator')?.setValue(condition);\r\n  }\r\n\r\n  copyRule(): void {\r\n    \r\n  }\r\n\r\n  createField(field: FormlyFieldConfig): FormlyFieldConfig {\r\n    if(field.type === 'dynamic-type' || field.type === '') {\r\n      field.type = 'sa-input';\r\n      field.props.type = 'text';\r\n      field.props.placeholder = 'Enter/Select';\r\n    }\r\n    return field;\r\n  }\r\n\r\n  getFilterType(subField: FormlyFieldConfig): string | undefined {\r\n    const parentValue = get(subField, 'formControl.parent.value');\r\n    if (!parentValue) return undefined;\r\n  \r\n    // Look through all keys of the parent value\r\n    for (const key of keys(parentValue)) {\r\n      const val = parentValue[key];\r\n      if (val && typeof val === 'object' && 'filterType' in val) {\r\n        return val.filterType;\r\n      }\r\n    }\r\n    return undefined;\r\n  }\r\n\r\n\r\n\r\n  // reset(): void {\r\n  //   this.formControl.reset();\r\n  // }\r\n\r\n  /**\r\n   * Duplicates the row at the given index.\r\n   * A deep–clone of that row’s model is inserted right after the original row.\r\n   */\r\n  // Helper method to backup all date range fields\r\n  private backupAllDateRangeFields(): any[] {\r\n    const backups: any[] = [];\r\n    \r\n    for (let i = 0; i < this.field.fieldGroup.length; i++) {\r\n      const rowFieldGroup = this.field.fieldGroup[i];\r\n      const filterValueField = rowFieldGroup?.fieldGroup?.find((f: any) => f.key === 'filterValue');\r\n      \r\n      if (filterValueField?.props?.['showDateRange']) {\r\n        backups[i] = {\r\n          props: { ...filterValueField.props },\r\n          fieldGroup: filterValueField.fieldGroup ? filterValueField.fieldGroup.map((field: any) => ({\r\n            key: field.key,\r\n            type: field.type,\r\n            className: field.className,\r\n            props: { ...field.props }\r\n            // Note: NOT backing up formControl - let Angular manage it\r\n          })) : null,\r\n          type: filterValueField.type\r\n          // Note: NOT backing up formControl - let Angular manage it\r\n        };\r\n      }\r\n    }\r\n    \r\n    console.log('🎯 Backed up', backups.filter(b => b).length, 'date range fields');\r\n    return backups;\r\n  }\r\n\r\n  // Helper method to restore all date range fields\r\n  private restoreAllDateRangeFields(backups: any[]): void {\r\n    let restoredCount = 0;\r\n    \r\n    for (let i = 0; i < backups.length; i++) {\r\n      if (backups[i]) {\r\n        const rowFieldGroup = this.field.fieldGroup[i];\r\n        const filterValueField = rowFieldGroup?.fieldGroup?.find((f: any) => f.key === 'filterValue');\r\n        \r\n        if (filterValueField) {\r\n          console.log(`🎯 Restoring row ${i} - before:`, {\r\n            hasShowDateRange: !!filterValueField.props?.['showDateRange'],\r\n            hasFieldGroup: !!filterValueField.fieldGroup,\r\n            fieldGroupLength: filterValueField.fieldGroup?.length\r\n          });\r\n          \r\n          filterValueField.props = backups[i].props;\r\n          filterValueField.fieldGroup = backups[i].fieldGroup;\r\n          filterValueField.type = backups[i].type;\r\n          \r\n          // Reconnect FormControls to fieldGroup items if it's a FormGroup\r\n          if (filterValueField.formControl && filterValueField.fieldGroup) {\r\n            const formGroup = filterValueField.formControl;\r\n            if (formGroup.get) { // Check if it's a FormGroup\r\n              filterValueField.fieldGroup.forEach((field: any) => {\r\n                if (field.key) {\r\n                  field.formControl = formGroup.get(field.key);\r\n                }\r\n              });\r\n              console.log(`🎯 Reconnected FormControls for row ${i} fieldGroup`);\r\n            }\r\n          }\r\n          \r\n          console.log(`🎯 Restoring row ${i} - after:`, {\r\n            hasShowDateRange: !!filterValueField.props?.['showDateRange'],\r\n            hasFieldGroup: !!filterValueField.fieldGroup,\r\n            fieldGroupLength: filterValueField.fieldGroup?.length\r\n          });\r\n          \r\n          restoredCount++;\r\n        } else {\r\n          console.log(`❌ Could not find filterValueField for row ${i}`);\r\n        }\r\n      }\r\n    }\r\n    console.log(`🎯 Restored ${restoredCount} date range fields`);\r\n  }\r\n\r\n  // Override add method to protect existing date range fields\r\n  override add(i?: number, initialModel?: any): void {\r\n    console.log('🎯 Add operation started - protecting existing date range fields');\r\n    \r\n    // BACKUP ALL DATE RANGE FIELDS BEFORE add() operation\r\n    const allDateRangeBackups = this.backupAllDateRangeFields();\r\n    \r\n    // Call the parent add method\r\n    super.add(i, initialModel);\r\n    \r\n    // RESTORE ALL DATE RANGE FIELDS after add() with multiple attempts\r\n    setTimeout(() => {\r\n      console.log('🎯 First restoration attempt');\r\n      this.restoreAllDateRangeFields(allDateRangeBackups);\r\n    }, 0);\r\n    \r\n    setTimeout(() => {\r\n      console.log('🎯 Second restoration attempt (after handleOperatorChange)');\r\n      this.restoreAllDateRangeFields(allDateRangeBackups);\r\n    }, 100);\r\n    \r\n    setTimeout(() => {\r\n      console.log('🎯 Final restoration attempt');\r\n      this.restoreAllDateRangeFields(allDateRangeBackups);\r\n    }, 300);\r\n  }\r\n\r\n  copy(index: number): void {\r\n    const rowModel = this.formControl.at(index)?.value;\r\n    if (!rowModel) return;\r\n\r\n    const clonedModel = JSON.parse(JSON.stringify(rowModel));\r\n\r\n    // Carry the value over to the new row so the onInit hook can restore it.\r\n    if (clonedModel.filterValue !== undefined) {\r\n      clonedModel._copiedValue = clonedModel.filterValue;\r\n    }\r\n\r\n    // If copying the first row (index 0), the new row (index 1) should have a logical operator\r\n    // Set default logical operator to 'and' for the new row\r\n    if (index === 0) {\r\n      clonedModel.logicalOperator = 'and';\r\n    }\r\n\r\n    // Determine if this is a date range before any modifications\r\n    const copiedValue = clonedModel._copiedValue;\r\n    const isDateRange = clonedModel.operators === 'BETWEEN' && \r\n                       clonedModel.attribute.filterType === 'DATE' &&\r\n                       copiedValue && typeof copiedValue === 'object' &&\r\n                       copiedValue.startDate && copiedValue.endDate;\r\n\r\n    // BACKUP ALL DATE RANGE FIELDS BEFORE add() operation\r\n    const allDateRangeBackups = this.backupAllDateRangeFields();\r\n\r\n    this.add(index + 1, clonedModel);\r\n\r\n    // IMMEDIATELY RESTORE ALL DATE RANGE FIELDS after add()\r\n    this.restoreAllDateRangeFields(allDateRangeBackups);\r\n\r\n    // Force rebuild of the filterValue field by triggering attribute change\r\n    setTimeout(() => {\r\n      const newRowForm = this.formControl.at(index + 1) as FormGroup;\r\n      const attrCtrl = newRowForm?.get('attribute');\r\n      const operatorCtrl = newRowForm?.get('operators');\r\n      const filterValueCtrl = newRowForm?.get('filterValue');\r\n      \r\n      if (attrCtrl && operatorCtrl && filterValueCtrl && clonedModel.attribute) {\r\n        \r\n        console.log('🎯 Copy: isDateRange =', isDateRange, 'copiedValue =', copiedValue);\r\n        \r\n        // For date range, set up everything without triggering change events\r\n        if (isDateRange) {\r\n          console.log('🎯 Copy: Setting up date range without events');\r\n          \r\n          // Set attribute and operator without events\r\n          attrCtrl.setValue(clonedModel.attribute, { emitEvent: false });\r\n          operatorCtrl.setValue(clonedModel.operators, { emitEvent: false });\r\n          \r\n          // Create FormGroup directly\r\n          const dateRangeGroup = new FormGroup({\r\n            startDate: new FormControl(copiedValue.startDate),\r\n            endDate: new FormControl(copiedValue.endDate)\r\n          });\r\n          \r\n          // Replace the FormControl with FormGroup\r\n          newRowForm.setControl('filterValue', dateRangeGroup);\r\n          \r\n          // Find the filterValue field in the new row and update its structure\r\n          const newRowFieldGroup = this.field.fieldGroup[index + 1];\r\n          const filterValueField = newRowFieldGroup?.fieldGroup?.find((f: any) => f.key === 'filterValue');\r\n          \r\n          if (filterValueField) {\r\n            // Update field structure to show fieldGroup (two date pickers)\r\n            filterValueField.props = { \r\n              ...filterValueField.props, \r\n              showDateRange: true,\r\n              attribute: clonedModel.attribute,\r\n              hasDynamicType: true\r\n            };\r\n            \r\n            // Create fieldGroup structure\r\n            filterValueField.fieldGroup = [\r\n              {\r\n                key: 'startDate',\r\n                type: 'datepicker',\r\n                className: 'query-builder-form-wrapper flex-2',\r\n                formControl: dateRangeGroup.get('startDate'),\r\n                props: {\r\n                  placeholder: 'Start date',\r\n                  label: 'Start Date'\r\n                }\r\n              },\r\n              {\r\n                key: 'endDate',\r\n                type: 'datepicker', \r\n                className: 'query-builder-form-wrapper flex-2',\r\n                formControl: dateRangeGroup.get('endDate'),\r\n                props: {\r\n                  placeholder: 'End date',\r\n                  label: 'End Date'\r\n                }\r\n              }\r\n            ];\r\n            \r\n            // Clear the type since we're using fieldGroup\r\n            filterValueField.type = null;\r\n            \r\n            // Update the formControl reference (cast to any to bypass readonly)\r\n            (filterValueField as any).formControl = dateRangeGroup;\r\n            \r\n            console.log('🎯 Copy: Date range setup complete', dateRangeGroup.value);\r\n          }\r\n        } else {\r\n          // For single values, use normal approach\r\n          attrCtrl.setValue(null, { emitEvent: false });\r\n          operatorCtrl.setValue(null, { emitEvent: false });\r\n          \r\n          setTimeout(() => {\r\n            attrCtrl.setValue(clonedModel.attribute, { emitEvent: true });\r\n            \r\n            setTimeout(() => {\r\n              operatorCtrl.setValue(clonedModel.operators, { emitEvent: true });\r\n              \r\n              setTimeout(() => {\r\n                if (copiedValue !== undefined) {\r\n                  filterValueCtrl.setValue(copiedValue);\r\n                }\r\n              }, 50);\r\n            }, 50);\r\n          }, 10);\r\n        }\r\n      }\r\n    });\r\n  }\r\n  \r\n  reset() {\r\n    this.field.fieldGroup.splice(0, this.field.fieldGroup.length); // clear\r\n    this.add(); // start fresh with one row\r\n  }\r\n\r\n}\r\n","<div class=\"sa-query-builder-container\">\r\n  <div class=\"query-builder-body\">\r\n    @for(subField of field.fieldGroup; track subField.id || $index) {\r\n    <div class=\"row align-items-baseline query-row\" [@rowAnimation]>\r\n     \r\n      @if ($index === 0) {\r\n      <span class=\"where-text flex-1\">Where</span>\r\n      }\r\n      @for (innerField of subField.fieldGroup; track innerField.key || $index) {\r\n        \r\n       @if(!!innerField.props['hasDynamicType']){\r\n       <!-- This is a single field -->\r\n      @switch (getFilterType(innerField)) {\r\n        \r\n      @case ('STRING') {\r\n      <formly-field [field]=\"innerField\"></formly-field>\r\n      }\r\n      @case ('NUMBER') {\r\n      <formly-field [field]=\"innerField\"></formly-field>\r\n      }\r\n      @case ('BOOLEAN') {\r\n      <formly-field [field]=\"innerField\"></formly-field>\r\n      }\r\n      @case ('CURRENCY') {\r\n      <formly-field [field]=\"innerField\"></formly-field>\r\n      }\r\n      @case ('DATE') {\r\n        {{innerField.props?.['showDateRange'] || false}}\r\n      @if(innerField.props?.['showDateRange']){\r\n        <!-- Date Range Mode - Two Date Pickers -->\r\n        @for(dateField of innerField.fieldGroup; track dateField.key) {\r\n          <mat-form-field [appearance]=\"'outline'\" class=\"query-builder-form-wrapper flex-2\">\r\n            <mat-label>{{dateField.key === 'startDate' ? 'Start Date' : 'End Date'}}</mat-label>\r\n            <input matInput [matDatepicker]=\"dateField.key === 'startDate' ? startPicker : endPicker\"\r\n              [placeholder]=\"dateField.key === 'startDate' ? 'Select start date' : 'Select end date'\"\r\n              [formControl]=\"$any(dateField.formControl)\" \r\n              [formlyAttributes]=\"dateField\" />\r\n            <mat-datepicker-toggle matSuffix\r\n              [for]=\"dateField.key === 'startDate' ? startPicker : endPicker\"></mat-datepicker-toggle>\r\n          </mat-form-field>\r\n        }\r\n        <mat-datepicker #startPicker></mat-datepicker>\r\n        <mat-datepicker #endPicker></mat-datepicker>\r\n      }\r\n      @else{\r\n      <mat-form-field [appearance]=\"innerField.props?.['appearance'] || 'outline'\"\r\n        class=\"query-builder-form-wrapper flex-2\">\r\n        <mat-label>{{innerField.key === 'endDate' ? 'End Date' : (innerField.props?.placeholder || 'Select\r\n          date')}}</mat-label>\r\n        <input matInput [matDatepicker]=\"picker\" [placeholder]=\"innerField.props?.placeholder || 'Select date'\"\r\n          [formControl]=\"$any(innerField.formControl)\" [formlyAttributes]=\"innerField\"\r\n          [required]=\"!!innerField.props?.required\" />\r\n        <mat-datepicker-toggle matSuffix [for]=\"picker\"></mat-datepicker-toggle>\r\n        <mat-datepicker #picker></mat-datepicker>\r\n      </mat-form-field>\r\n      }\r\n      }\r\n      @case ('SELECT') {\r\n      <formly-field [field]=\"createField(innerField)\"></formly-field>\r\n      }\r\n      @default {\r\n      <formly-field [field]=\"createField(innerField)\"></formly-field>\r\n      }\r\n\r\n      }\r\n      }@else{\r\n      <formly-field class=\"col\" [field]=\"innerField\">\r\n      </formly-field>\r\n      }\r\n      }\r\n      @if(props?.['showCopyButton']){\r\n      <sa-icon icon=\"copyOutlined\" class=\"copy-icon pointer\" (click)=\"copy($index)\"></sa-icon>\r\n      }\r\n      <sa-icon icon=\"deleteIcon\" class=\"delete-icon pointer\" (click)=\"remove($index)\"></sa-icon>\r\n    </div>\r\n    }\r\n\r\n    <!-- Buttons -->\r\n    <div class=\"rule-button-container\">\r\n      <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\r\n        [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\" type=\"outline\"\r\n        size=\"medium\">\r\n      </sa-button>\r\n      <span class=\"filters-count\">({{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters)</span>\r\n      @if(props?.['showResetButton']){\r\n      <sa-button iconPosition=\"left\" text=\"Reset\" (onClickEvent)=\"reset()\" type=\"primary\" size=\"medium\">\r\n      </sa-button>\r\n      }\r\n    </div>\r\n  </div>\r\n</div>"]}
339
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-builder-formly.component.js","sourceRoot":"","sources":["../../../../../projects/component-library/src/lib/query-builder-formly/query-builder-formly.component.ts","../../../../../projects/component-library/src/lib/query-builder-formly/query-builder-formly.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, OnInit } from '@angular/core';\nimport { FieldArrayType, FormlyFieldConfig } from '@ngx-formly/core';\nimport { CommonModule } from '@angular/common'; \nimport { FormlyModule } from '@ngx-formly/core';\nimport { FormGroup, FormControl, ReactiveFormsModule } from '@angular/forms';\nimport { get, keys } from 'lodash';\nimport { ButtonComponent } from '../button/button.component';\nimport { IconComponent } from '../icon/icon.component';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatSelectModule } from '@angular/material/select';\nimport { MatNativeDateModule, MatOptionModule } from '@angular/material/core';\nimport { MatIconModule } from '@angular/material/icon';\nimport { trigger, state, style, transition, animate, query, stagger } from '@angular/animations';\nimport { MatDatepickerModule } from '@angular/material/datepicker';\nimport { MatTooltipModule } from '@angular/material/tooltip';\n\n\n@Component({\n  selector: 'sa-query-builder-formly',\n  templateUrl: './query-builder-formly.component.html',\n  styleUrls: ['./query-builder-formly.component.scss'],\n  standalone: true,\n  providers: [],\n  animations: [\n    trigger('rowAnimation', [\n      transition(':enter', [\n        style({ \n          opacity: 0, \n          transform: 'translateY(-10px) scale(0.95)',\n          height: '0px',\n          overflow: 'hidden'\n        }),\n        animate('300ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({ \n          opacity: 1, \n          transform: 'translateY(0) scale(1)',\n          height: '*'\n        }))\n      ]),\n      transition(':leave', [\n        style({ \n          opacity: 1, \n          transform: 'translateY(0) scale(1)',\n          height: '*'\n        }),\n        animate('250ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({ \n          opacity: 0, \n          transform: 'translateY(-10px) scale(0.95)',\n          height: '0px',\n          overflow: 'hidden'\n        }))\n      ])\n    ])\n  ],\n  imports: [\n    CommonModule,\n    ReactiveFormsModule,\n    FormlyModule,\n    ButtonComponent,\n    IconComponent,\n    MatFormFieldModule,\n    MatInputModule,\n    MatSelectModule,\n    MatOptionModule,\n    MatIconModule,\n    MatDatepickerModule,\n    MatNativeDateModule,\n    MatTooltipModule\n  ]\n})\nexport class QueryBuilderFormlyComponent extends FieldArrayType implements OnInit {\n  \n  ngOnInit(): void {\n    \n    \n  }\n  \n  getRowCondition(index: number): 'and' | 'or' {\n      if (index === 0) return 'and'; \n      \n      const ruleGroup = this.formControl.at(index) as FormGroup;\n      return ruleGroup.get('logicalOperator')?.value || 'and';\n    }\n\n    /**\n   * Sets the condition for a specific row\n   * @param index The index of the row\n   * @param condition The condition to set (and/or)\n   */\n  setRowCondition(index: number, condition: 'and' | 'or'): void {\n    if (index === 0) return; // First row doesn't have a logical operator\n    \n    const ruleGroup = this.formControl.at(index) as FormGroup;\n    ruleGroup.get('logicalOperator')?.setValue(condition);\n  }\n\n  copyRule(): void {\n    \n  }\n\n  createField(field: FormlyFieldConfig): FormlyFieldConfig {\n    if(field.type === 'dynamic-type' || field.type === '') {\n      field.type = 'sa-input';\n      field.props.type = 'text';\n      field.props.placeholder = 'Enter/Select';\n    }\n    return field;\n  }\n\n\n\n\n  // reset(): void {\n  //   this.formControl.reset();\n  // }\n\n  /**\n   * Duplicates the row at the given index.\n   * A deep–clone of that row’s model is inserted right after the original row.\n   */\n  // Helper method to backup all date range fields\n  private backupAllDateRangeFields(): any[] {\n    const backups: any[] = [];\n    \n    for (let i = 0; i < this.field.fieldGroup.length; i++) {\n      const rowFieldGroup = this.field.fieldGroup[i];\n      const filterValueField = rowFieldGroup?.fieldGroup?.find((f: any) => f.key === 'filterValue');\n      \n      if (filterValueField?.props?.['showDateRange']) {\n        backups[i] = {\n          props: { ...filterValueField.props },\n          fieldGroup: filterValueField.fieldGroup ? filterValueField.fieldGroup.map((field: any) => ({\n            key: field.key,\n            type: field.type,\n            className: field.className,\n            props: { ...field.props }\n            // Note: NOT backing up formControl - let Angular manage it\n          })) : null,\n          type: filterValueField.type\n          // Note: NOT backing up formControl - let Angular manage it\n        };\n      }\n    }\n    \n    console.log('🎯 Backed up', backups.filter(b => b).length, 'date range fields');\n    return backups;\n  }\n\n  // Helper method to restore all date range fields\n  private restoreAllDateRangeFields(backups: any[]): void {\n    let restoredCount = 0;\n    \n    for (let i = 0; i < backups.length; i++) {\n      if (backups[i]) {\n        const rowFieldGroup = this.field.fieldGroup[i];\n        const filterValueField = rowFieldGroup?.fieldGroup?.find((f: any) => f.key === 'filterValue');\n        \n        if (filterValueField) {\n          console.log(`🎯 Restoring row ${i} - before:`, {\n            hasShowDateRange: !!filterValueField.props?.['showDateRange'],\n            hasFieldGroup: !!filterValueField.fieldGroup,\n            fieldGroupLength: filterValueField.fieldGroup?.length\n          });\n          \n          filterValueField.props = backups[i].props;\n          filterValueField.fieldGroup = backups[i].fieldGroup;\n          filterValueField.type = backups[i].type;\n          \n          // Reconnect FormControls to fieldGroup items if it's a FormGroup\n          if (filterValueField.formControl && filterValueField.fieldGroup) {\n            const formGroup = filterValueField.formControl;\n            if (formGroup.get) { // Check if it's a FormGroup\n              filterValueField.fieldGroup.forEach((field: any) => {\n                if (field.key) {\n                  field.formControl = formGroup.get(field.key);\n                }\n              });\n              console.log(`🎯 Reconnected FormControls for row ${i} fieldGroup`);\n            }\n          }\n          \n          console.log(`🎯 Restoring row ${i} - after:`, {\n            hasShowDateRange: !!filterValueField.props?.['showDateRange'],\n            hasFieldGroup: !!filterValueField.fieldGroup,\n            fieldGroupLength: filterValueField.fieldGroup?.length\n          });\n          \n          restoredCount++;\n        } else {\n          console.log(`❌ Could not find filterValueField for row ${i}`);\n        }\n      }\n    }\n    console.log(`🎯 Restored ${restoredCount} date range fields`);\n  }\n\n  // Override add method to protect existing date range fields\n  // override add(i?: number, initialModel?: any): void {\n  //   console.log('🎯 Add operation started - protecting existing date range fields');\n    \n  //   // BACKUP ALL DATE RANGE FIELDS BEFORE add() operation\n  //   const allDateRangeBackups = this.backupAllDateRangeFields();\n    \n  //   // Call the parent add method\n  //   super.add(i, initialModel);\n    \n  //   // RESTORE ALL DATE RANGE FIELDS after add() with multiple attempts\n  //   setTimeout(() => {\n  //     console.log('🎯 First restoration attempt');\n  //     this.restoreAllDateRangeFields(allDateRangeBackups);\n  //   }, 0);\n    \n  //   setTimeout(() => {\n  //     console.log('🎯 Second restoration attempt (after handleOperatorChange)');\n  //     this.restoreAllDateRangeFields(allDateRangeBackups);\n  //   }, 100);\n    \n  //   setTimeout(() => {\n  //     console.log('🎯 Final restoration attempt');\n  //     this.restoreAllDateRangeFields(allDateRangeBackups);\n  //   }, 300);\n  // }\n  isRowValidForCopy(index: number): boolean {\n    const rowModel = this.formControl.at(index)?.value;\n    if (!rowModel || typeof rowModel !== 'object') return false;\n\n    // Check if required fields are filled: attribute, operators, and either filterValue or filterValueDateRange\n    const hasAttribute = rowModel.attribute && typeof rowModel.attribute === 'object';\n    const hasOperator = rowModel.operators && rowModel.operators !== '';\n    \n    // Check filterValue (for non-date-range fields) - can be empty string, but must be defined\n    const hasFilterValue = 'filterValue' in rowModel && \n                           rowModel.filterValue !== undefined && \n                           rowModel.filterValue !== null;\n    \n    // Check filterValueDateRange (for date range fields with BETWEEN operator)\n    const hasDateRangeValue = rowModel.filterValueDateRange && \n                               typeof rowModel.filterValueDateRange === 'object' &&\n                               (rowModel.filterValueDateRange.start || rowModel.filterValueDateRange.end);\n    \n    // Row is valid if it has attribute, operator, and at least one value field\n    return hasAttribute && hasOperator && (hasFilterValue || hasDateRangeValue);\n  }\n  \n  reset() {\n    this.field.fieldGroup.splice(0, this.field.fieldGroup.length); // clear\n    this.add(); // start fresh with one row\n  }\n\n  getFilterType(subField: FormlyFieldConfig): string | undefined {\n    const parentValue = get(subField, 'formControl.parent.value');\n    if (!parentValue) return undefined;\n\n    // Look through all keys of the parent value\n    for (const key of keys(parentValue)) {\n      const val = parentValue[key];\n      if (val && typeof val === 'object' && 'filterType' in val) {\n        return get(val, 'filterType');\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Helper method to restore a simple field value\n   */\n  private restoreSimpleField(formGroup: FormGroup, fieldName: string, value: any, backupKey?: string): void {\n    const ctrl = formGroup?.get(fieldName);\n    if (ctrl && value !== undefined) {\n      if (backupKey && !formGroup.value[backupKey]) {\n        formGroup.patchValue({ [backupKey]: value });\n      }\n      ctrl.setValue(value);\n    }\n  }\n\n  /**\n   * Helper method to restore a fieldGroup with nested controls (like date range)\n   */\n  private restoreFieldGroup(formGroup: FormGroup, fieldName: string, value: any): void {\n    const ctrl = formGroup?.get(fieldName);\n    if (ctrl && value && typeof value === 'object') {\n      Object.keys(value).forEach(key => {\n        if (value[key] !== undefined) {\n          ctrl.get(key)?.setValue(value[key]);\n        }\n      });\n    }\n  }\n\n  /**\n   * Duplicates the row at the given index.\n   * A deep–clone of that row's model is inserted right after the original row.\n   */\n  copy(index: number): void {\n    const rowModel = this.formControl.at(index)?.value;\n    if (!rowModel) return;\n\n    const clonedModel = JSON.parse(JSON.stringify(rowModel));\n\n    // Define fields to copy with their backup keys\n    const fieldsToCopy = [\n      { field: 'filterValue', backupKey: '_copiedValue' },\n      { field: 'filterValueDateRange', backupKey: '_copiedValueDateRange' }\n    ];\n\n    // Store copied values for restoration\n    fieldsToCopy.forEach(({ field, backupKey }) => {\n      if (clonedModel[field] !== undefined) {\n        clonedModel[backupKey] = clonedModel[field];\n      }\n    });\n\n    // If copying the first row (index 0), the new row (index 1) should have a logical operator\n    if (index === 0) {\n      clonedModel.logicalOperator = 'and';\n    }\n\n    this.add(index + 1, clonedModel);\n\n    // Force rebuild of fields by triggering attribute change\n    setTimeout(() => {\n      const newRowForm = this.formControl.at(index + 1) as FormGroup;\n      const attrCtrl = newRowForm?.get('attribute');\n\n      if (!attrCtrl || !clonedModel.attribute) return;\n\n      // Store all values to restore\n      const valuesToRestore = {\n        operators: clonedModel.operators,\n        filterValue: clonedModel._copiedValue,\n        filterValueDateRange: clonedModel._copiedValueDateRange\n      };\n\n      // Temporarily clear the attribute to force a change event\n      attrCtrl.setValue(null, { emitEvent: false });\n\n      // Restore attribute to trigger field type rebuild\n      setTimeout(() => {\n        attrCtrl.setValue(clonedModel.attribute, { emitEvent: true });\n\n        // Restore operator after attribute is set\n        setTimeout(() => {\n          this.restoreSimpleField(newRowForm, 'operators', valuesToRestore.operators);\n        }, 30);\n\n        // Restore all value fields after rebuild\n        setTimeout(() => {\n          // Restore simple filterValue\n          this.restoreSimpleField(\n            newRowForm, \n            'filterValue', \n            valuesToRestore.filterValue, \n            '_copiedValue'\n          );\n          \n          // Restore fieldGroup filterValueDateRange\n          this.restoreFieldGroup(\n            newRowForm, \n            'filterValueDateRange', \n            valuesToRestore.filterValueDateRange\n          );\n        }, 100);\n      }, 10);\n    });\n  }\n\n}\n","<div class=\"sa-query-builder-container\">\n  <div class=\"query-builder-body\">\n      @for(row of field.fieldGroup; track row.id || $index; let last = $last) {\n\n      <div class=\"row align-items-baseline query-row\">\n          @if ($index === 0) {\n          <span class=\"where-text flex-1\">Where the:</span>\n          }\n          @for (ruleField of row.fieldGroup; track ruleField.key || $index) {\n          @if(!!ruleField?.props?.['hasDynamicType']){\n          @switch (getFilterType(ruleField)) {\n          @case ('STRING') {\n          <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n              [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n          }\n          @case ('NUMBER') {\n          <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n              [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n          }\n          @case ('BOOLEAN') {\n          <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n              [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n          }\n          @case ('CURRENCY') {\n          <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n              [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n          }\n          @case ('DATE') {\n          <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n              [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n          }\n          @case ('SELECT') {\n          <formly-field [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n              [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n          }\n          @default {\n          <formly-field [field]=\"createField(ruleField)\" matTooltip=\"Select a filter first\"\n              [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n          }\n          }\n          }@else{\n              <formly-field class=\"col\" [field]=\"ruleField\" matTooltip=\"Select a filter first\"\n              [matTooltipDisabled]=\"!ruleField.formControl?.disabled\"></formly-field>\n          }\n          }\n          <div class=\"rule-icon-container\">\n              @if(props?.['showCopyButton']){\n              <sa-icon icon=\"copyOutlined\" size=\"20\" matTooltip=\"Copy\" class=\"copy-icon pointer {{(field?.fieldGroup?.length === (props?.['maxRows'] || 5)) || !isRowValidForCopy($index) ? 'disabled-part' : ''}}\" (click)=\"isRowValidForCopy($index) && copy($index)\"></sa-icon>\n              }\n              @if(props?.['showDeleteRowButton']){\n              <sa-icon icon=\"deleteIcon\" size=\"20\" matTooltip=\"Delete\" class=\"delete-icon pointer {{field?.fieldGroup?.length === (1) ? 'disabled-part' : ''}}\" (click)=\"remove($index)\"></sa-icon>\n              }\n          </div>\n      </div>\n      @if(!last || (last && props?.['showLastDivider'])){\n      <div class=\"segment-divider\"></div>\n      }\n      }\n\n      <!-- Buttons -->\n      <div class=\"rule-button-container\">\n          @if(props?.['showAddFilterButton']){\n          <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\n              [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n              type=\"outline\" size=\"medium\">\n          </sa-button>\n          }\n          @if(props?.['showCount']){\n          <span class=\"filters-count\">{{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters</span>\n          }\n          @if(props?.['showResetButton']){\n          <sa-button iconPosition=\"left\" text=\"Reset\" (onClickEvent)=\"reset()\" type=\"primary\" size=\"medium\">\n          </sa-button>\n          }\n      </div>\n  </div>\n</div>"]}