@sarasanalytics-com/design-system 0.0.159 → 0.0.161
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +24 -24
- package/esm2022/interfaces/avatar-interface.mjs +1 -1
- package/esm2022/interfaces/button-interface.mjs +1 -1
- package/esm2022/interfaces/card-carousel-interface.mjs +1 -1
- package/esm2022/interfaces/chip-interface.mjs +1 -1
- package/esm2022/interfaces/form-layout.interface.mjs +1 -1
- package/esm2022/interfaces/grid-interface.mjs +1 -1
- package/esm2022/interfaces/guide-card-interface.mjs +1 -1
- package/esm2022/interfaces/header-interface.mjs +1 -1
- package/esm2022/interfaces/icon-interface.mjs +1 -1
- package/esm2022/interfaces/layout-section-interface.mjs +1 -1
- package/esm2022/interfaces/left-nav-interface.mjs +1 -1
- package/esm2022/interfaces/menu.interface.mjs +1 -1
- package/esm2022/interfaces/message-banner.mjs +1 -1
- package/esm2022/interfaces/option-interface.mjs +1 -1
- package/esm2022/interfaces/scrolling-card-interface.mjs +1 -1
- package/esm2022/interfaces/select-interface.mjs +1 -1
- package/esm2022/interfaces/status-dot.interface.mjs +1 -1
- package/esm2022/interfaces/tab-interface.mjs +1 -1
- package/esm2022/interfaces/toast-interface.mjs +1 -1
- package/esm2022/interfaces/typography-animation-interface.mjs +1 -1
- package/esm2022/lib/accordion/accordion.component.mjs +3 -3
- package/esm2022/lib/avatar/avatar.component.mjs +3 -3
- package/esm2022/lib/button/button.component.mjs +3 -3
- package/esm2022/lib/calendar-header/calendar-header.component.mjs +3 -3
- package/esm2022/lib/card/card-body/card-body.component.mjs +1 -1
- package/esm2022/lib/card/card-custom-header/card-custom-header.component.mjs +1 -1
- package/esm2022/lib/card/card-footer-actions/card-footer-actions.component.mjs +3 -3
- package/esm2022/lib/card/card-icon/card-icon.component.mjs +1 -1
- package/esm2022/lib/card/card-title-actions/card-title-actions.component.mjs +1 -1
- package/esm2022/lib/card/card.component.mjs +3 -3
- package/esm2022/lib/card/checkbox-card/checkbox-card.component.mjs +3 -3
- package/esm2022/lib/card/guide-card/guide-card.component.mjs +3 -3
- package/esm2022/lib/card/menu-card/menu-card.component.mjs +3 -3
- package/esm2022/lib/card/thumbnail-card/thumbnail-card.component.mjs +3 -3
- package/esm2022/lib/card-carousel/card-carousel.component.mjs +3 -3
- package/esm2022/lib/categories-nav/categories-nav.component.mjs +3 -3
- package/esm2022/lib/checkbox/checkbox.component.mjs +3 -3
- package/esm2022/lib/chips/chips.component.mjs +3 -3
- package/esm2022/lib/component-library.component.mjs +9 -9
- package/esm2022/lib/component-library.service.mjs +1 -1
- package/esm2022/lib/dashboard-loader/dashboard-loader.component.mjs +1 -1
- package/esm2022/lib/data-grid/data-grid.component.mjs +4 -3
- package/esm2022/lib/datepicker/datepicker.component.mjs +3 -3
- package/esm2022/lib/dialog/dialog.component.mjs +3 -3
- package/esm2022/lib/dropdown/category-dropdown/category-dropdown.component.mjs +3 -3
- package/esm2022/lib/dropdown/ng-select/ng-select.component.mjs +3 -3
- package/esm2022/lib/filter/filter.component.mjs +3 -3
- package/esm2022/lib/form-input/form-input.component.mjs +3 -3
- package/esm2022/lib/form-select/form-select.component.mjs +3 -3
- package/esm2022/lib/grid-cell/grid-cell.component.mjs +16 -11
- package/esm2022/lib/header/header.component.mjs +3 -3
- package/esm2022/lib/icon/icon.component.mjs +3 -3
- package/esm2022/lib/icon/icon.service.mjs +1 -1
- package/esm2022/lib/layout-section/layout-section.component.mjs +3 -3
- package/esm2022/lib/left-nav/left-nav.component.mjs +3 -3
- package/esm2022/lib/list/list.component.mjs +3 -3
- package/esm2022/lib/menu/menu-list/menu-item.component.mjs +3 -3
- package/esm2022/lib/menu/menu.component.mjs +3 -3
- package/esm2022/lib/menu/menu.directive.mjs +1 -1
- package/esm2022/lib/message-banner/message-banner.component.mjs +3 -3
- package/esm2022/lib/message-banner-v2/message-banner-v2.component.mjs +4 -4
- package/esm2022/lib/mini-card/mini-card.component.mjs +3 -3
- package/esm2022/lib/page-layout/page-layout.component.mjs +3 -3
- package/esm2022/lib/progress-bar/progress-bar.component.mjs +3 -3
- package/esm2022/lib/query-builder/query-builder-demo.component.mjs +45 -45
- package/esm2022/lib/query-builder/query-builder.component.mjs +3 -3
- package/esm2022/lib/query-builder/query-builder.service.mjs +1 -1
- package/esm2022/lib/query-builder-formly/query-builder-formly.component.mjs +185 -23
- package/esm2022/lib/query-builder-textarea/query-builder-textarea-demo.component.mjs +39 -39
- package/esm2022/lib/query-builder-textarea/query-builder-textarea.component.mjs +3 -3
- package/esm2022/lib/radio-button/radio-button.component.mjs +3 -3
- package/esm2022/lib/scrolling-cards/scrolling-cards.component.mjs +3 -3
- package/esm2022/lib/skeleton/skeleton-base.component.mjs +1 -1
- package/esm2022/lib/skeleton/skeleton-container.component.mjs +9 -9
- package/esm2022/lib/skeleton/skeleton-loader.component.mjs +193 -193
- package/esm2022/lib/skeleton/skeleton-presets.mjs +1 -1
- package/esm2022/lib/skeleton/skeleton-shapes.component.mjs +1 -1
- package/esm2022/lib/spinner/spinner.component.mjs +1 -1
- package/esm2022/lib/status-dot/status-dot.component.mjs +17 -17
- package/esm2022/lib/stepper/stepper.component.mjs +3 -3
- package/esm2022/lib/tabs/tabs.component.mjs +3 -3
- package/esm2022/lib/toast/toast.component.mjs +3 -3
- package/esm2022/lib/tool-tip/tool-tip.component.mjs +3 -3
- package/esm2022/lib/typography-animation/typography-animation.component.mjs +3 -3
- package/esm2022/utils/validators.mjs +1 -1
- package/fesm2022/sarasanalytics-com-design-system.mjs +589 -430
- package/fesm2022/sarasanalytics-com-design-system.mjs.map +1 -1
- package/interfaces/grid-interface.d.ts +4 -0
- package/lib/data-grid/data-grid.component.d.ts +6 -0
- package/lib/query-builder-formly/query-builder-formly.component.d.ts +3 -0
- package/package.json +1 -1
- package/styles/styles.css +432 -432
|
@@ -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,
|
|
107
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicXVlcnktYnVpbGRlci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY29tcG9uZW50LWxpYnJhcnkvc3JjL2xpYi9xdWVyeS1idWlsZGVyL3F1ZXJ5LWJ1aWxkZXIuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQU0zQyxNQUFNLE9BQU8sbUJBQW1CO0lBRTlCLGdCQUFnQixDQUFDO0lBRWpCOzs7O09BSUc7SUFDSCxhQUFhLENBQUMsS0FBaUI7UUFDN0IsSUFBSSxDQUFDLEtBQUssSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDdkQsT0FBTyxFQUFFLENBQUM7UUFDWixDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUMsS0FBSzthQUNmLEdBQUcsQ0FBQyxDQUFDLElBQVMsRUFBRSxFQUFFO1lBQ2pCLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUNuQixPQUFPLElBQUksSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFrQixDQUFDLEdBQUcsQ0FBQztZQUN2RCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sT0FBTyxHQUFHLElBQUksQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDeEQsQ0FBQztRQUNILENBQUMsQ0FBQzthQUNELElBQUksQ0FBQyxJQUFJLEtBQUssQ0FBQyxTQUFTLENBQUMsV0FBVyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ2hELENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsV0FBVyxDQUFDLEtBQWlCO1FBQzNCLElBQUksQ0FBQyxLQUFLLElBQUksQ0FBQyxLQUFLLENBQUMsS0FBSyxJQUFJLEtBQUssQ0FBQyxLQUFLLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQ3ZELE9BQU8sRUFBRSxDQUFDO1FBQ1osQ0FBQztRQUVELE9BQU87WUFDTCxTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVM7WUFDMUIsS0FBSyxFQUFFLEtBQUssQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBUyxFQUFFLEVBQUU7Z0JBQ25DLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO29CQUNuQixPQUFPLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBa0IsQ0FBQyxDQUFDO2dCQUM5QyxDQUFDO3FCQUFNLENBQUM7b0JBQ04sT0FBTzt3QkFDTCxLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUs7d0JBQ2pCLFFBQVEsRUFBRSxJQUFJLENBQUMsUUFBUTt3QkFDdkIsS0FBSyxFQUFFLElBQUksQ0FBQyxLQUFLO3FCQUNsQixDQUFDO2dCQUNKLENBQUM7WUFDSCxDQUFDLENBQUM7U0FDSCxDQUFDO0lBQ0osQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxZQUFZLENBQUMsS0FBaUI7UUFDNUIsSUFBSSxDQUFDLEtBQUssSUFBSSxDQUFDLEtBQUssQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDLEtBQUssQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDdkQsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUMsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDLElBQVMsRUFBRSxFQUFFO1lBQ3JDLElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUNuQixPQUFPLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBa0IsQ0FBQyxDQUFDO1lBQy9DLENBQUM7aUJBQU0sQ0FBQztnQkFDTixPQUFPLElBQUksQ0FBQyxLQUFLLElBQUksSUFBSSxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLFNBQVMsSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLElBQUksQ0FBQztZQUN4RixDQUFDO1FBQ0gsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILGdCQUFnQixDQUFDLFlBQTBCLEtBQUs7UUFDOUMsT0FBTztZQUNMLFNBQVM7WUFDVCxLQUFLLEVBQUUsRUFBRTtTQUNWLENBQUM7SUFDSixDQUFDO0lBRUQ7Ozs7OztPQU1HO0lBQ0gsaUJBQWlCLENBQUMsS0FBYSxFQUFFLFFBQWdCLEVBQUUsS0FBVTtRQUMzRCxPQUFPO1lBQ0wsU0FBUyxFQUFFLEtBQUs7WUFDaEIsS0FBSyxFQUFFO2dCQUNMO29CQUNFLEtBQUs7b0JBQ0wsUUFBUTtvQkFDUixLQUFLO2lCQUNOO2FBQ0Y7U0FDRixDQUFDO0lBQ0osQ0FBQzs4R0FwR1UsbUJBQW1CO2tIQUFuQixtQkFBbUIsY0FGbEIsTUFBTTs7MkZBRVAsbUJBQW1CO2tCQUgvQixVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFF1ZXJ5R3JvdXAsIFF1ZXJ5UnVsZSB9IGZyb20gJy4vcXVlcnktYnVpbGRlci5jb21wb25lbnQnO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290J1xufSlcbmV4cG9ydCBjbGFzcyBRdWVyeUJ1aWxkZXJTZXJ2aWNlIHtcbiAgXG4gIGNvbnN0cnVjdG9yKCkgeyB9XG5cbiAgLyoqXG4gICAqIENvbnZlcnRzIGEgcXVlcnkgZ3JvdXAgdG8gYSBodW1hbi1yZWFkYWJsZSBzdHJpbmdcbiAgICogQHBhcmFtIHF1ZXJ5IFRoZSBxdWVyeSBncm91cCB0byBjb252ZXJ0XG4gICAqIEByZXR1cm5zIEEgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBxdWVyeVxuICAgKi9cbiAgcXVlcnlUb1N0cmluZyhxdWVyeTogUXVlcnlHcm91cCk6IHN0cmluZyB7XG4gICAgaWYgKCFxdWVyeSB8fCAhcXVlcnkucnVsZXMgfHwgcXVlcnkucnVsZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gJyc7XG4gICAgfVxuXG4gICAgcmV0dXJuIHF1ZXJ5LnJ1bGVzXG4gICAgICAubWFwKChydWxlOiBhbnkpID0+IHtcbiAgICAgICAgaWYgKHJ1bGUuY29uZGl0aW9uKSB7XG4gICAgICAgICAgcmV0dXJuIGAoJHt0aGlzLnF1ZXJ5VG9TdHJpbmcocnVsZSBhcyBRdWVyeUdyb3VwKX0pYDtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICByZXR1cm4gYCR7cnVsZS5maWVsZH0gJHtydWxlLm9wZXJhdG9yfSAke3J1bGUudmFsdWV9YDtcbiAgICAgICAgfVxuICAgICAgfSlcbiAgICAgIC5qb2luKGAgJHtxdWVyeS5jb25kaXRpb24udG9VcHBlckNhc2UoKX0gYCk7XG4gIH1cblxuICAvKipcbiAgICogQ29udmVydHMgYSBxdWVyeSBncm91cCB0byBhIEpTT04gb2JqZWN0IHRoYXQgY2FuIGJlIHVzZWQgaW4gQVBJIHJlcXVlc3RzXG4gICAqIEBwYXJhbSBxdWVyeSBUaGUgcXVlcnkgZ3JvdXAgdG8gY29udmVydFxuICAgKiBAcmV0dXJucyBBIEpTT04gb2JqZWN0IHJlcHJlc2VudGF0aW9uIG9mIHRoZSBxdWVyeVxuICAgKi9cbiAgcXVlcnlUb0pzb24ocXVlcnk6IFF1ZXJ5R3JvdXApOiBhbnkge1xuICAgIGlmICghcXVlcnkgfHwgIXF1ZXJ5LnJ1bGVzIHx8IHF1ZXJ5LnJ1bGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIHt9O1xuICAgIH1cblxuICAgIHJldHVybiB7XG4gICAgICBjb25kaXRpb246IHF1ZXJ5LmNvbmRpdGlvbixcbiAgICAgIHJ1bGVzOiBxdWVyeS5ydWxlcy5tYXAoKHJ1bGU6IGFueSkgPT4ge1xuICAgICAgICBpZiAocnVsZS5jb25kaXRpb24pIHtcbiAgICAgICAgICByZXR1cm4gdGhpcy5xdWVyeVRvSnNvbihydWxlIGFzIFF1ZXJ5R3JvdXApO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBmaWVsZDogcnVsZS5maWVsZCxcbiAgICAgICAgICAgIG9wZXJhdG9yOiBydWxlLm9wZXJhdG9yLFxuICAgICAgICAgICAgdmFsdWU6IHJ1bGUudmFsdWVcbiAgICAgICAgICB9O1xuICAgICAgICB9XG4gICAgICB9KVxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogVmFsaWRhdGVzIGEgcXVlcnkgZ3JvdXAgdG8gZW5zdXJlIGFsbCBydWxlcyBoYXZlIHZhbGlkIHZhbHVlc1xuICAgKiBAcGFyYW0gcXVlcnkgVGhlIHF1ZXJ5IGdyb3VwIHRvIHZhbGlkYXRlXG4gICAqIEByZXR1cm5zIFRydWUgaWYgdGhlIHF1ZXJ5IGlzIHZhbGlkLCBmYWxzZSBvdGhlcndpc2VcbiAgICovXG4gIGlzUXVlcnlWYWxpZChxdWVyeTogUXVlcnlHcm91cCk6IGJvb2xlYW4ge1xuICAgIGlmICghcXVlcnkgfHwgIXF1ZXJ5LnJ1bGVzIHx8IHF1ZXJ5LnJ1bGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cblxuICAgIHJldHVybiBxdWVyeS5ydWxlcy5ldmVyeSgocnVsZTogYW55KSA9PiB7XG4gICAgICBpZiAocnVsZS5jb25kaXRpb24pIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuaXNRdWVyeVZhbGlkKHJ1bGUgYXMgUXVlcnlHcm91cCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZXR1cm4gcnVsZS5maWVsZCAmJiBydWxlLm9wZXJhdG9yICYmIHJ1bGUudmFsdWUgIT09IHVuZGVmaW5lZCAmJiBydWxlLnZhbHVlICE9PSBudWxsO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZXMgYW4gZW1wdHkgcXVlcnkgZ3JvdXBcbiAgICogQHBhcmFtIGNvbmRpdGlvbiBUaGUgY29uZGl0aW9uIHRvIHVzZSAoYW5kL29yKVxuICAgKiBAcmV0dXJucyBBbiBlbXB0eSBxdWVyeSBncm91cFxuICAgKi9cbiAgY3JlYXRlRW1wdHlRdWVyeShjb25kaXRpb246ICdhbmQnIHwgJ29yJyA9ICdhbmQnKTogUXVlcnlHcm91cCB7XG4gICAgcmV0dXJuIHtcbiAgICAgIGNvbmRpdGlvbixcbiAgICAgIHJ1bGVzOiBbXVxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlcyBhIHNpbXBsZSBxdWVyeSB3aXRoIGEgc2luZ2xlIHJ1bGVcbiAgICogQHBhcmFtIGZpZWxkIFRoZSBmaWVsZCB0byBmaWx0ZXIgb25cbiAgICogQHBhcmFtIG9wZXJhdG9yIFRoZSBvcGVyYXRvciB0byB1c2VcbiAgICogQHBhcmFtIHZhbHVlIFRoZSB2YWx1ZSB0byBmaWx0ZXIgYnlcbiAgICogQHJldHVybnMgQSBxdWVyeSBncm91cCB3aXRoIGEgc2luZ2xlIHJ1bGVcbiAgICovXG4gIGNyZWF0ZVNpbXBsZVF1ZXJ5KGZpZWxkOiBzdHJpbmcsIG9wZXJhdG9yOiBzdHJpbmcsIHZhbHVlOiBhbnkpOiBRdWVyeUdyb3VwIHtcbiAgICByZXR1cm4ge1xuICAgICAgY29uZGl0aW9uOiAnYW5kJyxcbiAgICAgIHJ1bGVzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICBmaWVsZCxcbiAgICAgICAgICBvcGVyYXRvcixcbiAgICAgICAgICB2YWx1ZVxuICAgICAgICB9XG4gICAgICBdXG4gICAgfTtcbiAgfVxufVxuIl19
|
|
@@ -2,17 +2,23 @@ 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 { ReactiveFormsModule } from '@angular/forms';
|
|
5
|
+
import { FormGroup, FormControl, 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';
|
|
9
9
|
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
10
|
+
import { MatInputModule } from '@angular/material/input';
|
|
10
11
|
import { MatSelectModule } from '@angular/material/select';
|
|
11
|
-
import { MatOptionModule } from '@angular/material/core';
|
|
12
|
+
import { MatNativeDateModule, MatOptionModule } from '@angular/material/core';
|
|
12
13
|
import { MatIconModule } from '@angular/material/icon';
|
|
13
14
|
import { trigger, style, transition, animate } from '@angular/animations';
|
|
15
|
+
import { MatDatepickerModule } from '@angular/material/datepicker';
|
|
14
16
|
import * as i0 from "@angular/core";
|
|
15
|
-
import * as i1 from "@
|
|
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";
|
|
16
22
|
export class QueryBuilderFormlyComponent extends FieldArrayType {
|
|
17
23
|
ngOnInit() {
|
|
18
24
|
}
|
|
@@ -63,6 +69,93 @@ export class QueryBuilderFormlyComponent extends FieldArrayType {
|
|
|
63
69
|
* Duplicates the row at the given index.
|
|
64
70
|
* A deep–clone of that row’s model is inserted right after the original row.
|
|
65
71
|
*/
|
|
72
|
+
// Helper method to backup all date range fields
|
|
73
|
+
backupAllDateRangeFields() {
|
|
74
|
+
const backups = [];
|
|
75
|
+
for (let i = 0; i < this.field.fieldGroup.length; i++) {
|
|
76
|
+
const rowFieldGroup = this.field.fieldGroup[i];
|
|
77
|
+
const filterValueField = rowFieldGroup?.fieldGroup?.find((f) => f.key === 'filterValue');
|
|
78
|
+
if (filterValueField?.props?.['showDateRange']) {
|
|
79
|
+
backups[i] = {
|
|
80
|
+
props: { ...filterValueField.props },
|
|
81
|
+
fieldGroup: filterValueField.fieldGroup ? filterValueField.fieldGroup.map((field) => ({
|
|
82
|
+
key: field.key,
|
|
83
|
+
type: field.type,
|
|
84
|
+
className: field.className,
|
|
85
|
+
props: { ...field.props }
|
|
86
|
+
// Note: NOT backing up formControl - let Angular manage it
|
|
87
|
+
})) : null,
|
|
88
|
+
type: filterValueField.type
|
|
89
|
+
// Note: NOT backing up formControl - let Angular manage it
|
|
90
|
+
};
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
console.log('🎯 Backed up', backups.filter(b => b).length, 'date range fields');
|
|
94
|
+
return backups;
|
|
95
|
+
}
|
|
96
|
+
// Helper method to restore all date range fields
|
|
97
|
+
restoreAllDateRangeFields(backups) {
|
|
98
|
+
let restoredCount = 0;
|
|
99
|
+
for (let i = 0; i < backups.length; i++) {
|
|
100
|
+
if (backups[i]) {
|
|
101
|
+
const rowFieldGroup = this.field.fieldGroup[i];
|
|
102
|
+
const filterValueField = rowFieldGroup?.fieldGroup?.find((f) => f.key === 'filterValue');
|
|
103
|
+
if (filterValueField) {
|
|
104
|
+
console.log(`🎯 Restoring row ${i} - before:`, {
|
|
105
|
+
hasShowDateRange: !!filterValueField.props?.['showDateRange'],
|
|
106
|
+
hasFieldGroup: !!filterValueField.fieldGroup,
|
|
107
|
+
fieldGroupLength: filterValueField.fieldGroup?.length
|
|
108
|
+
});
|
|
109
|
+
filterValueField.props = backups[i].props;
|
|
110
|
+
filterValueField.fieldGroup = backups[i].fieldGroup;
|
|
111
|
+
filterValueField.type = backups[i].type;
|
|
112
|
+
// Reconnect FormControls to fieldGroup items if it's a FormGroup
|
|
113
|
+
if (filterValueField.formControl && filterValueField.fieldGroup) {
|
|
114
|
+
const formGroup = filterValueField.formControl;
|
|
115
|
+
if (formGroup.get) { // Check if it's a FormGroup
|
|
116
|
+
filterValueField.fieldGroup.forEach((field) => {
|
|
117
|
+
if (field.key) {
|
|
118
|
+
field.formControl = formGroup.get(field.key);
|
|
119
|
+
}
|
|
120
|
+
});
|
|
121
|
+
console.log(`🎯 Reconnected FormControls for row ${i} fieldGroup`);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
console.log(`🎯 Restoring row ${i} - after:`, {
|
|
125
|
+
hasShowDateRange: !!filterValueField.props?.['showDateRange'],
|
|
126
|
+
hasFieldGroup: !!filterValueField.fieldGroup,
|
|
127
|
+
fieldGroupLength: filterValueField.fieldGroup?.length
|
|
128
|
+
});
|
|
129
|
+
restoredCount++;
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
console.log(`❌ Could not find filterValueField for row ${i}`);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
console.log(`🎯 Restored ${restoredCount} date range fields`);
|
|
137
|
+
}
|
|
138
|
+
// 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);
|
|
158
|
+
}
|
|
66
159
|
copy(index) {
|
|
67
160
|
const rowModel = this.formControl.at(index)?.value;
|
|
68
161
|
if (!rowModel)
|
|
@@ -77,29 +170,95 @@ export class QueryBuilderFormlyComponent extends FieldArrayType {
|
|
|
77
170
|
if (index === 0) {
|
|
78
171
|
clonedModel.logicalOperator = 'and';
|
|
79
172
|
}
|
|
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();
|
|
80
181
|
this.add(index + 1, clonedModel);
|
|
182
|
+
// IMMEDIATELY RESTORE ALL DATE RANGE FIELDS after add()
|
|
183
|
+
this.restoreAllDateRangeFields(allDateRangeBackups);
|
|
81
184
|
// Force rebuild of the filterValue field by triggering attribute change
|
|
82
185
|
setTimeout(() => {
|
|
83
186
|
const newRowForm = this.formControl.at(index + 1);
|
|
84
187
|
const attrCtrl = newRowForm?.get('attribute');
|
|
188
|
+
const operatorCtrl = newRowForm?.get('operators');
|
|
85
189
|
const filterValueCtrl = newRowForm?.get('filterValue');
|
|
86
|
-
if (attrCtrl && filterValueCtrl && clonedModel.attribute) {
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
//
|
|
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 });
|
|
95
250
|
setTimeout(() => {
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
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
|
+
}
|
|
103
262
|
}
|
|
104
263
|
});
|
|
105
264
|
}
|
|
@@ -108,7 +267,7 @@ export class QueryBuilderFormlyComponent extends FieldArrayType {
|
|
|
108
267
|
this.add(); // start fresh with one row
|
|
109
268
|
}
|
|
110
269
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.4", ngImport: i0, type: QueryBuilderFormlyComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
|
|
111
|
-
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\">\
|
|
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\">\n <div class=\"query-builder-body\">\n @for(subField of field.fieldGroup; track subField.id || $index) {\n <div class=\"row align-items-baseline query-row\" [@rowAnimation]>\n \n @if ($index === 0) {\n <span class=\"where-text flex-1\">Where</span>\n }\n @for (innerField of subField.fieldGroup; track innerField.key || $index) {\n \n @if(!!innerField.props['hasDynamicType']){\n <!-- This is a single field -->\n @switch (getFilterType(innerField)) {\n \n @case ('STRING') {\n <formly-field [field]=\"innerField\"></formly-field>\n }\n @case ('NUMBER') {\n <formly-field [field]=\"innerField\"></formly-field>\n }\n @case ('BOOLEAN') {\n <formly-field [field]=\"innerField\"></formly-field>\n }\n @case ('CURRENCY') {\n <formly-field [field]=\"innerField\"></formly-field>\n }\n @case ('DATE') {\n {{innerField.props?.['showDateRange'] || false}}\n @if(innerField.props?.['showDateRange']){\n <!-- Date Range Mode - Two Date Pickers -->\n @for(dateField of innerField.fieldGroup; track dateField.key) {\n <mat-form-field [appearance]=\"'outline'\" class=\"query-builder-form-wrapper flex-2\">\n <mat-label>{{dateField.key === 'startDate' ? 'Start Date' : 'End Date'}}</mat-label>\n <input matInput [matDatepicker]=\"dateField.key === 'startDate' ? startPicker : endPicker\"\n [placeholder]=\"dateField.key === 'startDate' ? 'Select start date' : 'Select end date'\"\n [formControl]=\"$any(dateField.formControl)\" \n [formlyAttributes]=\"dateField\" />\n <mat-datepicker-toggle matSuffix\n [for]=\"dateField.key === 'startDate' ? startPicker : endPicker\"></mat-datepicker-toggle>\n </mat-form-field>\n }\n <mat-datepicker #startPicker></mat-datepicker>\n <mat-datepicker #endPicker></mat-datepicker>\n }\n @else{\n <mat-form-field [appearance]=\"innerField.props?.['appearance'] || 'outline'\"\n class=\"query-builder-form-wrapper flex-2\">\n <mat-label>{{innerField.key === 'endDate' ? 'End Date' : (innerField.props?.placeholder || 'Select\n date')}}</mat-label>\n <input matInput [matDatepicker]=\"picker\" [placeholder]=\"innerField.props?.placeholder || 'Select date'\"\n [formControl]=\"$any(innerField.formControl)\" [formlyAttributes]=\"innerField\"\n [required]=\"!!innerField.props?.required\" />\n <mat-datepicker-toggle matSuffix [for]=\"picker\"></mat-datepicker-toggle>\n <mat-datepicker #picker></mat-datepicker>\n </mat-form-field>\n }\n }\n @case ('SELECT') {\n <formly-field [field]=\"createField(innerField)\"></formly-field>\n }\n @default {\n <formly-field [field]=\"createField(innerField)\"></formly-field>\n }\n\n }\n }@else{\n <formly-field class=\"col\" [field]=\"innerField\">\n </formly-field>\n }\n }\n @if(props?.['showCopyButton']){\n <sa-icon icon=\"copyOutlined\" class=\"copy-icon pointer\" (click)=\"copy($index)\"></sa-icon>\n }\n <sa-icon icon=\"deleteIcon\" class=\"delete-icon pointer\" (click)=\"remove($index)\"></sa-icon>\n </div>\n }\n\n <!-- Buttons -->\n <div class=\"rule-button-container\">\n <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\" type=\"outline\"\n size=\"medium\">\n </sa-button>\n <span class=\"filters-count\">({{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters)</span>\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)}\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", "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: [
|
|
112
271
|
trigger('rowAnimation', [
|
|
113
272
|
transition(':enter', [
|
|
114
273
|
style({
|
|
@@ -177,9 +336,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImpor
|
|
|
177
336
|
ButtonComponent,
|
|
178
337
|
IconComponent,
|
|
179
338
|
MatFormFieldModule,
|
|
339
|
+
MatInputModule,
|
|
180
340
|
MatSelectModule,
|
|
181
341
|
MatOptionModule,
|
|
182
|
-
MatIconModule
|
|
183
|
-
|
|
342
|
+
MatIconModule,
|
|
343
|
+
MatDatepickerModule,
|
|
344
|
+
MatNativeDateModule,
|
|
345
|
+
], template: "<div class=\"sa-query-builder-container\">\n <div class=\"query-builder-body\">\n @for(subField of field.fieldGroup; track subField.id || $index) {\n <div class=\"row align-items-baseline query-row\" [@rowAnimation]>\n \n @if ($index === 0) {\n <span class=\"where-text flex-1\">Where</span>\n }\n @for (innerField of subField.fieldGroup; track innerField.key || $index) {\n \n @if(!!innerField.props['hasDynamicType']){\n <!-- This is a single field -->\n @switch (getFilterType(innerField)) {\n \n @case ('STRING') {\n <formly-field [field]=\"innerField\"></formly-field>\n }\n @case ('NUMBER') {\n <formly-field [field]=\"innerField\"></formly-field>\n }\n @case ('BOOLEAN') {\n <formly-field [field]=\"innerField\"></formly-field>\n }\n @case ('CURRENCY') {\n <formly-field [field]=\"innerField\"></formly-field>\n }\n @case ('DATE') {\n {{innerField.props?.['showDateRange'] || false}}\n @if(innerField.props?.['showDateRange']){\n <!-- Date Range Mode - Two Date Pickers -->\n @for(dateField of innerField.fieldGroup; track dateField.key) {\n <mat-form-field [appearance]=\"'outline'\" class=\"query-builder-form-wrapper flex-2\">\n <mat-label>{{dateField.key === 'startDate' ? 'Start Date' : 'End Date'}}</mat-label>\n <input matInput [matDatepicker]=\"dateField.key === 'startDate' ? startPicker : endPicker\"\n [placeholder]=\"dateField.key === 'startDate' ? 'Select start date' : 'Select end date'\"\n [formControl]=\"$any(dateField.formControl)\" \n [formlyAttributes]=\"dateField\" />\n <mat-datepicker-toggle matSuffix\n [for]=\"dateField.key === 'startDate' ? startPicker : endPicker\"></mat-datepicker-toggle>\n </mat-form-field>\n }\n <mat-datepicker #startPicker></mat-datepicker>\n <mat-datepicker #endPicker></mat-datepicker>\n }\n @else{\n <mat-form-field [appearance]=\"innerField.props?.['appearance'] || 'outline'\"\n class=\"query-builder-form-wrapper flex-2\">\n <mat-label>{{innerField.key === 'endDate' ? 'End Date' : (innerField.props?.placeholder || 'Select\n date')}}</mat-label>\n <input matInput [matDatepicker]=\"picker\" [placeholder]=\"innerField.props?.placeholder || 'Select date'\"\n [formControl]=\"$any(innerField.formControl)\" [formlyAttributes]=\"innerField\"\n [required]=\"!!innerField.props?.required\" />\n <mat-datepicker-toggle matSuffix [for]=\"picker\"></mat-datepicker-toggle>\n <mat-datepicker #picker></mat-datepicker>\n </mat-form-field>\n }\n }\n @case ('SELECT') {\n <formly-field [field]=\"createField(innerField)\"></formly-field>\n }\n @default {\n <formly-field [field]=\"createField(innerField)\"></formly-field>\n }\n\n }\n }@else{\n <formly-field class=\"col\" [field]=\"innerField\">\n </formly-field>\n }\n }\n @if(props?.['showCopyButton']){\n <sa-icon icon=\"copyOutlined\" class=\"copy-icon pointer\" (click)=\"copy($index)\"></sa-icon>\n }\n <sa-icon icon=\"deleteIcon\" class=\"delete-icon pointer\" (click)=\"remove($index)\"></sa-icon>\n </div>\n }\n\n <!-- Buttons -->\n <div class=\"rule-button-container\">\n <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\" type=\"outline\"\n size=\"medium\">\n </sa-button>\n <span class=\"filters-count\">({{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters)</span>\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)}\n"] }]
|
|
184
346
|
}] });
|
|
185
|
-
//# 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, 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 { MatSelectModule } from '@angular/material/select';\r\nimport { 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\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    MatSelectModule,\r\n    MatOptionModule,\r\n    MatIconModule    \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  // 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  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    this.add(index + 1, clonedModel);\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 filterValueCtrl = newRowForm?.get('filterValue');\r\n      \r\n      if (attrCtrl && filterValueCtrl && clonedModel.attribute) {\r\n        // Store the copied value in the form model before triggering rebuild\r\n        const copiedValue = clonedModel._copiedValue;\r\n        \r\n        // Temporarily clear the attribute to force a change event\r\n        attrCtrl.setValue(null, { emitEvent: false });\r\n        \r\n        // Then restore it to trigger the field type rebuild\r\n        setTimeout(() => {\r\n          attrCtrl.setValue(clonedModel.attribute, { emitEvent: true });\r\n          \r\n          // Ensure the copied value is still available after the rebuild\r\n          setTimeout(() => {\r\n            if (copiedValue !== undefined && !newRowForm.value._copiedValue) {\r\n              // Re-add the copied value if it was lost during rebuild\r\n              newRowForm.patchValue({ _copiedValue: copiedValue });\r\n              filterValueCtrl.setValue(copiedValue);\r\n            }\r\n          }, 50);\r\n        }, 10);\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        @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          @if(!!innerField.props['hasDynamicType']){\r\n            @switch (getFilterType(innerField)) {\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\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          }@else{\r\n            <formly-field\r\n            class=\"col\"\r\n            [field]=\"innerField\"\r\n          >\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\"\r\n      (onClickEvent)=\"add()\"\r\n      [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\r\n      type=\"outline\" 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\"\r\n      (onClickEvent)=\"reset()\"\r\n      type=\"primary\" size=\"medium\">\r\n      </sa-button>\r\n      }\r\n    </div>    \r\n    </div>\r\n</div>"]}
|
|
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';\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';\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  ]\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  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 val.filterType;\n      }\n    }\n    return undefined;\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\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    // Carry the value over to the new row so the onInit hook can restore it.\n    if (clonedModel.filterValue !== undefined) {\n      clonedModel._copiedValue = clonedModel.filterValue;\n    }\n\n    // If copying the first row (index 0), the new row (index 1) should have a logical operator\n    // Set default logical operator to 'and' for the new row\n    if (index === 0) {\n      clonedModel.logicalOperator = 'and';\n    }\n\n    // Determine if this is a date range before any modifications\n    const copiedValue = clonedModel._copiedValue;\n    const isDateRange = clonedModel.operators === 'BETWEEN' && \n                       clonedModel.attribute.filterType === 'DATE' &&\n                       copiedValue && typeof copiedValue === 'object' &&\n                       copiedValue.startDate && copiedValue.endDate;\n\n    // BACKUP ALL DATE RANGE FIELDS BEFORE add() operation\n    const allDateRangeBackups = this.backupAllDateRangeFields();\n\n    this.add(index + 1, clonedModel);\n\n    // IMMEDIATELY RESTORE ALL DATE RANGE FIELDS after add()\n    this.restoreAllDateRangeFields(allDateRangeBackups);\n\n    // Force rebuild of the filterValue field by triggering attribute change\n    setTimeout(() => {\n      const newRowForm = this.formControl.at(index + 1) as FormGroup;\n      const attrCtrl = newRowForm?.get('attribute');\n      const operatorCtrl = newRowForm?.get('operators');\n      const filterValueCtrl = newRowForm?.get('filterValue');\n      \n      if (attrCtrl && operatorCtrl && filterValueCtrl && clonedModel.attribute) {\n        \n        console.log('🎯 Copy: isDateRange =', isDateRange, 'copiedValue =', copiedValue);\n        \n        // For date range, set up everything without triggering change events\n        if (isDateRange) {\n          console.log('🎯 Copy: Setting up date range without events');\n          \n          // Set attribute and operator without events\n          attrCtrl.setValue(clonedModel.attribute, { emitEvent: false });\n          operatorCtrl.setValue(clonedModel.operators, { emitEvent: false });\n          \n          // Create FormGroup directly\n          const dateRangeGroup = new FormGroup({\n            startDate: new FormControl(copiedValue.startDate),\n            endDate: new FormControl(copiedValue.endDate)\n          });\n          \n          // Replace the FormControl with FormGroup\n          newRowForm.setControl('filterValue', dateRangeGroup);\n          \n          // Find the filterValue field in the new row and update its structure\n          const newRowFieldGroup = this.field.fieldGroup[index + 1];\n          const filterValueField = newRowFieldGroup?.fieldGroup?.find((f: any) => f.key === 'filterValue');\n          \n          if (filterValueField) {\n            // Update field structure to show fieldGroup (two date pickers)\n            filterValueField.props = { \n              ...filterValueField.props, \n              showDateRange: true,\n              attribute: clonedModel.attribute,\n              hasDynamicType: true\n            };\n            \n            // Create fieldGroup structure\n            filterValueField.fieldGroup = [\n              {\n                key: 'startDate',\n                type: 'datepicker',\n                className: 'query-builder-form-wrapper flex-2',\n                formControl: dateRangeGroup.get('startDate'),\n                props: {\n                  placeholder: 'Start date',\n                  label: 'Start Date'\n                }\n              },\n              {\n                key: 'endDate',\n                type: 'datepicker', \n                className: 'query-builder-form-wrapper flex-2',\n                formControl: dateRangeGroup.get('endDate'),\n                props: {\n                  placeholder: 'End date',\n                  label: 'End Date'\n                }\n              }\n            ];\n            \n            // Clear the type since we're using fieldGroup\n            filterValueField.type = null;\n            \n            // Update the formControl reference (cast to any to bypass readonly)\n            (filterValueField as any).formControl = dateRangeGroup;\n            \n            console.log('🎯 Copy: Date range setup complete', dateRangeGroup.value);\n          }\n        } else {\n          // For single values, use normal approach\n          attrCtrl.setValue(null, { emitEvent: false });\n          operatorCtrl.setValue(null, { emitEvent: false });\n          \n          setTimeout(() => {\n            attrCtrl.setValue(clonedModel.attribute, { emitEvent: true });\n            \n            setTimeout(() => {\n              operatorCtrl.setValue(clonedModel.operators, { emitEvent: true });\n              \n              setTimeout(() => {\n                if (copiedValue !== undefined) {\n                  filterValueCtrl.setValue(copiedValue);\n                }\n              }, 50);\n            }, 50);\n          }, 10);\n        }\n      }\n    });\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}\n","<div class=\"sa-query-builder-container\">\n  <div class=\"query-builder-body\">\n    @for(subField of field.fieldGroup; track subField.id || $index) {\n    <div class=\"row align-items-baseline query-row\" [@rowAnimation]>\n     \n      @if ($index === 0) {\n      <span class=\"where-text flex-1\">Where</span>\n      }\n      @for (innerField of subField.fieldGroup; track innerField.key || $index) {\n        \n       @if(!!innerField.props['hasDynamicType']){\n       <!-- This is a single field -->\n      @switch (getFilterType(innerField)) {\n        \n      @case ('STRING') {\n      <formly-field [field]=\"innerField\"></formly-field>\n      }\n      @case ('NUMBER') {\n      <formly-field [field]=\"innerField\"></formly-field>\n      }\n      @case ('BOOLEAN') {\n      <formly-field [field]=\"innerField\"></formly-field>\n      }\n      @case ('CURRENCY') {\n      <formly-field [field]=\"innerField\"></formly-field>\n      }\n      @case ('DATE') {\n        {{innerField.props?.['showDateRange'] || false}}\n      @if(innerField.props?.['showDateRange']){\n        <!-- Date Range Mode - Two Date Pickers -->\n        @for(dateField of innerField.fieldGroup; track dateField.key) {\n          <mat-form-field [appearance]=\"'outline'\" class=\"query-builder-form-wrapper flex-2\">\n            <mat-label>{{dateField.key === 'startDate' ? 'Start Date' : 'End Date'}}</mat-label>\n            <input matInput [matDatepicker]=\"dateField.key === 'startDate' ? startPicker : endPicker\"\n              [placeholder]=\"dateField.key === 'startDate' ? 'Select start date' : 'Select end date'\"\n              [formControl]=\"$any(dateField.formControl)\" \n              [formlyAttributes]=\"dateField\" />\n            <mat-datepicker-toggle matSuffix\n              [for]=\"dateField.key === 'startDate' ? startPicker : endPicker\"></mat-datepicker-toggle>\n          </mat-form-field>\n        }\n        <mat-datepicker #startPicker></mat-datepicker>\n        <mat-datepicker #endPicker></mat-datepicker>\n      }\n      @else{\n      <mat-form-field [appearance]=\"innerField.props?.['appearance'] || 'outline'\"\n        class=\"query-builder-form-wrapper flex-2\">\n        <mat-label>{{innerField.key === 'endDate' ? 'End Date' : (innerField.props?.placeholder || 'Select\n          date')}}</mat-label>\n        <input matInput [matDatepicker]=\"picker\" [placeholder]=\"innerField.props?.placeholder || 'Select date'\"\n          [formControl]=\"$any(innerField.formControl)\" [formlyAttributes]=\"innerField\"\n          [required]=\"!!innerField.props?.required\" />\n        <mat-datepicker-toggle matSuffix [for]=\"picker\"></mat-datepicker-toggle>\n        <mat-datepicker #picker></mat-datepicker>\n      </mat-form-field>\n      }\n      }\n      @case ('SELECT') {\n      <formly-field [field]=\"createField(innerField)\"></formly-field>\n      }\n      @default {\n      <formly-field [field]=\"createField(innerField)\"></formly-field>\n      }\n\n      }\n      }@else{\n      <formly-field class=\"col\" [field]=\"innerField\">\n      </formly-field>\n      }\n      }\n      @if(props?.['showCopyButton']){\n      <sa-icon icon=\"copyOutlined\" class=\"copy-icon pointer\" (click)=\"copy($index)\"></sa-icon>\n      }\n      <sa-icon icon=\"deleteIcon\" class=\"delete-icon pointer\" (click)=\"remove($index)\"></sa-icon>\n    </div>\n    }\n\n    <!-- Buttons -->\n    <div class=\"rule-button-container\">\n      <sa-button icon=\"add\" iconPosition=\"left\" text=\"Add filter\" (onClickEvent)=\"add()\"\n        [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\" type=\"outline\"\n        size=\"medium\">\n      </sa-button>\n      <span class=\"filters-count\">({{field?.fieldGroup?.length}}/{{props?.['maxRows'] || 5}} filters)</span>\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>"]}
|