@sarasanalytics-com/design-system 0.0.155 → 0.0.157

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.
@@ -10,27 +10,17 @@ import { MatFormFieldModule } from '@angular/material/form-field';
10
10
  import { MatSelectModule } from '@angular/material/select';
11
11
  import { MatOptionModule } from '@angular/material/core';
12
12
  import { MatIconModule } from '@angular/material/icon';
13
+ import { trigger, style, transition, animate } from '@angular/animations';
13
14
  import * as i0 from "@angular/core";
14
15
  import * as i1 from "@ngx-formly/core";
15
16
  export class QueryBuilderFormlyComponent extends FieldArrayType {
16
- constructor() {
17
- super(...arguments);
18
- // reset(): void {
19
- // this.formControl.reset();
20
- // }
21
- /**
22
- * Duplicates the row at the given index.
23
- * A deep–clone of that row’s model is inserted right after the original row.
24
- */
25
- this.copied = false;
26
- }
27
17
  ngOnInit() {
28
18
  }
29
19
  getRowCondition(index) {
30
20
  if (index === 0)
31
21
  return 'and';
32
22
  const ruleGroup = this.formControl.at(index);
33
- return ruleGroup.get('condition')?.value || 'and';
23
+ return ruleGroup.get('logicalOperator')?.value || 'and';
34
24
  }
35
25
  /**
36
26
  * Sets the condition for a specific row
@@ -39,9 +29,9 @@ export class QueryBuilderFormlyComponent extends FieldArrayType {
39
29
  */
40
30
  setRowCondition(index, condition) {
41
31
  if (index === 0)
42
- return; // First row doesn't have a condition
32
+ return; // First row doesn't have a logical operator
43
33
  const ruleGroup = this.formControl.at(index);
44
- ruleGroup.get('condition')?.setValue(condition);
34
+ ruleGroup.get('logicalOperator')?.setValue(condition);
45
35
  }
46
36
  copyRule() {
47
37
  }
@@ -66,117 +56,121 @@ export class QueryBuilderFormlyComponent extends FieldArrayType {
66
56
  }
67
57
  return undefined;
68
58
  }
59
+ // reset(): void {
60
+ // this.formControl.reset();
61
+ // }
62
+ /**
63
+ * Duplicates the row at the given index.
64
+ * A deep–clone of that row’s model is inserted right after the original row.
65
+ */
69
66
  copy(index) {
70
67
  const rowModel = this.formControl.at(index)?.value;
71
68
  if (!rowModel)
72
69
  return;
73
- // Deep-clone model
74
- const cloned = JSON.parse(JSON.stringify(rowModel));
75
- // Insert row after the current one
76
- this.add(index + 1, cloned);
77
- // 👇 Force Formly to refresh the dynamic field
78
- const newRow = this.formControl.at(index + 1);
79
- const attr = newRow?.get('attribute')?.value;
80
- if (attr) {
81
- const attrCtrl = newRow?.get('attribute');
82
- attrCtrl.setValue(attrCtrl.value);
83
- // Re-trigger field type rebuild for filterValue
84
- const filterValueField = this.field.fieldGroup[index + 1].fieldGroup
85
- .find(f => f.key === 'filterValue');
86
- if (filterValueField) {
87
- const typeMap = {
88
- SELECT: 'sa-ng-select',
89
- STRING: 'sa-input',
90
- NUMBER: 'sa-input',
91
- DATE: 'sa-date-picker',
92
- CURRENCY: 'sa-currency-input',
93
- BOOLEAN: 'sa-ng-select',
94
- };
95
- const newType = typeMap[attr.filterType] || 'sa-input';
96
- filterValueField.type = newType;
97
- filterValueField.props = {
98
- ...this.createFieldProps(attr.filterType),
99
- hasDynamicType: true,
100
- disabled: false, // initial
101
- attribute: attr,
102
- };
103
- // 🔄 Notify Formly of type change
104
- filterValueField.options?.fieldChanges?.next({
105
- type: 'type',
106
- field: filterValueField,
107
- value: newType,
108
- });
109
- }
70
+ const clonedModel = JSON.parse(JSON.stringify(rowModel));
71
+ // Carry the value over to the new row so the onInit hook can restore it.
72
+ if (clonedModel.filterValue !== undefined) {
73
+ clonedModel._copiedValue = clonedModel.filterValue;
110
74
  }
111
- }
112
- createFieldProps(type) {
113
- switch (type) {
114
- case 'SELECT':
115
- return {
116
- placeholder: 'Select',
117
- multiple: false,
118
- bindLabel: 'label',
119
- bindValue: 'value',
120
- appearance: 'outline',
121
- isClearable: true,
122
- isSearchable: false,
123
- closeOnselect: true,
124
- required: true,
125
- disabled: false,
126
- };
127
- case 'STRING':
128
- return {
129
- placeholder: 'Enter value',
130
- appearance: 'outline',
131
- type: 'text'
132
- };
133
- case 'NUMBER':
134
- return {
135
- placeholder: 'Enter value',
136
- appearance: 'outline',
137
- type: 'number'
138
- };
139
- case 'DATE':
140
- return {
141
- placeholder: 'Select date',
142
- appearance: 'outline',
143
- type: 'date'
144
- };
145
- case 'CURRENCY':
146
- return {
147
- placeholder: 'Enter value',
148
- appearance: 'outline',
149
- };
150
- case 'BOOLEAN':
151
- return {
152
- placeholder: 'Enter value',
153
- appearance: 'outline',
154
- type: 'boolean',
155
- options: [
156
- { value: true, label: 'True' },
157
- { value: false, label: 'False' },
158
- ],
159
- bindLabel: 'label',
160
- bindValue: 'value',
161
- disabled: false,
162
- };
163
- default:
164
- return {
165
- placeholder: 'Enter value',
166
- appearance: 'outline',
167
- };
75
+ // If copying the first row (index 0), the new row (index 1) should have a logical operator
76
+ // Set default logical operator to 'and' for the new row
77
+ if (index === 0) {
78
+ clonedModel.logicalOperator = 'and';
168
79
  }
80
+ this.add(index + 1, clonedModel);
81
+ // Force rebuild of the filterValue field by triggering attribute change
82
+ setTimeout(() => {
83
+ const newRowForm = this.formControl.at(index + 1);
84
+ const attrCtrl = newRowForm?.get('attribute');
85
+ const filterValueCtrl = newRowForm?.get('filterValue');
86
+ if (attrCtrl && filterValueCtrl && clonedModel.attribute) {
87
+ // Store the copied value in the form model before triggering rebuild
88
+ const copiedValue = clonedModel._copiedValue;
89
+ // Temporarily clear the attribute to force a change event
90
+ attrCtrl.setValue(null, { emitEvent: false });
91
+ // Then restore it to trigger the field type rebuild
92
+ setTimeout(() => {
93
+ attrCtrl.setValue(clonedModel.attribute, { emitEvent: true });
94
+ // Ensure the copied value is still available after the rebuild
95
+ setTimeout(() => {
96
+ if (copiedValue !== undefined && !newRowForm.value._copiedValue) {
97
+ // Re-add the copied value if it was lost during rebuild
98
+ newRowForm.patchValue({ _copiedValue: copiedValue });
99
+ filterValueCtrl.setValue(copiedValue);
100
+ }
101
+ }, 50);
102
+ }, 10);
103
+ }
104
+ });
169
105
  }
170
106
  reset() {
171
107
  this.field.fieldGroup.splice(0, this.field.fieldGroup.length); // clear
172
108
  this.add(); // start fresh with one row
173
109
  }
174
110
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.4", ngImport: i0, type: QueryBuilderFormlyComponent, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
175
- 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 $index) {\n <div class=\"row align-items-baseline\">\n @if ($index === 0) {\n <span class=\"where-text flex-1\">Where</span>\n }\n @for (subField of subField.fieldGroup; track $index) {\n @if(!!subField.props['hasDynamicType']){\n @switch (getFilterType(subField)) {\n @case ('STRING') {\n <formly-field [field]=\"subField\"></formly-field>\n }\n @case ('NUMBER') {\n <formly-field [field]=\"subField\"></formly-field>\n }\n @case ('BOOLEAN') {\n <formly-field [field]=\"subField\"></formly-field>\n }\n @case ('CURRENCY') {\n <formly-field [field]=\"subField\"></formly-field>\n }\n @case ('DATE') {\n\n }\n @case ('SELECT') {\n <formly-field [field]=\"createField(subField)\"></formly-field>\n }\n @default {\n <formly-field [field]=\"createField(subField)\"></formly-field>\n }\n }\n }@else{\n <formly-field\n class=\"col\"\n [field]=\"subField\"\n >\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\"\n (onClickEvent)=\"add()\"\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n type=\"outline\" 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\"\n (onClickEvent)=\"reset()\"\n 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-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}\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", "icon", "size", "color", "iconPath", "iconUrl", "customClass", "href", "hrefTarget", "iconPosition"], outputs: ["onClickEvent"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatIconModule }] }); }
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\">\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 @if ($index === 0) {\n <span class=\"where-text flex-1\">Where</span>\n }\n @for (innerField of subField.fieldGroup; track innerField.key || $index) {\n @if(!!innerField.props['hasDynamicType']){\n @switch (getFilterType(innerField)) {\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\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 }@else{\n <formly-field\n class=\"col\"\n [field]=\"innerField\"\n >\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\"\n (onClickEvent)=\"add()\"\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n type=\"outline\" 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\"\n (onClickEvent)=\"reset()\"\n 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:hover{transform:translate(2px)}.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}.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: "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", "icon", "size", "color", "iconPath", "iconUrl", "customClass", "href", "hrefTarget", "iconPosition"], outputs: ["onClickEvent"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatOptionModule }, { kind: "ngmodule", type: MatIconModule }], animations: [
112
+ trigger('rowAnimation', [
113
+ transition(':enter', [
114
+ style({
115
+ opacity: 0,
116
+ transform: 'translateY(-10px) scale(0.95)',
117
+ height: '0px',
118
+ overflow: 'hidden'
119
+ }),
120
+ animate('300ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({
121
+ opacity: 1,
122
+ transform: 'translateY(0) scale(1)',
123
+ height: '*'
124
+ }))
125
+ ]),
126
+ transition(':leave', [
127
+ style({
128
+ opacity: 1,
129
+ transform: 'translateY(0) scale(1)',
130
+ height: '*'
131
+ }),
132
+ animate('250ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({
133
+ opacity: 0,
134
+ transform: 'translateY(-10px) scale(0.95)',
135
+ height: '0px',
136
+ overflow: 'hidden'
137
+ }))
138
+ ])
139
+ ])
140
+ ] }); }
176
141
  }
177
142
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImport: i0, type: QueryBuilderFormlyComponent, decorators: [{
178
143
  type: Component,
179
- args: [{ selector: 'sa-query-builder-formly', standalone: true, providers: [], imports: [
144
+ args: [{ selector: 'sa-query-builder-formly', standalone: true, providers: [], animations: [
145
+ trigger('rowAnimation', [
146
+ transition(':enter', [
147
+ style({
148
+ opacity: 0,
149
+ transform: 'translateY(-10px) scale(0.95)',
150
+ height: '0px',
151
+ overflow: 'hidden'
152
+ }),
153
+ animate('300ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({
154
+ opacity: 1,
155
+ transform: 'translateY(0) scale(1)',
156
+ height: '*'
157
+ }))
158
+ ]),
159
+ transition(':leave', [
160
+ style({
161
+ opacity: 1,
162
+ transform: 'translateY(0) scale(1)',
163
+ height: '*'
164
+ }),
165
+ animate('250ms cubic-bezier(0.4, 0.0, 0.2, 1)', style({
166
+ opacity: 0,
167
+ transform: 'translateY(-10px) scale(0.95)',
168
+ height: '0px',
169
+ overflow: 'hidden'
170
+ }))
171
+ ])
172
+ ])
173
+ ], imports: [
180
174
  CommonModule,
181
175
  ReactiveFormsModule,
182
176
  FormlyModule,
@@ -186,6 +180,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImpor
186
180
  MatSelectModule,
187
181
  MatOptionModule,
188
182
  MatIconModule
189
- ], template: "<div class=\"sa-query-builder-container\">\n <div class=\"query-builder-body\">\n @for(subField of field.fieldGroup; track $index) {\n <div class=\"row align-items-baseline\">\n @if ($index === 0) {\n <span class=\"where-text flex-1\">Where</span>\n }\n @for (subField of subField.fieldGroup; track $index) {\n @if(!!subField.props['hasDynamicType']){\n @switch (getFilterType(subField)) {\n @case ('STRING') {\n <formly-field [field]=\"subField\"></formly-field>\n }\n @case ('NUMBER') {\n <formly-field [field]=\"subField\"></formly-field>\n }\n @case ('BOOLEAN') {\n <formly-field [field]=\"subField\"></formly-field>\n }\n @case ('CURRENCY') {\n <formly-field [field]=\"subField\"></formly-field>\n }\n @case ('DATE') {\n\n }\n @case ('SELECT') {\n <formly-field [field]=\"createField(subField)\"></formly-field>\n }\n @default {\n <formly-field [field]=\"createField(subField)\"></formly-field>\n }\n }\n }@else{\n <formly-field\n class=\"col\"\n [field]=\"subField\"\n >\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\"\n (onClickEvent)=\"add()\"\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n type=\"outline\" 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\"\n (onClickEvent)=\"reset()\"\n 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-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}\n"] }]
183
+ ], 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 @if ($index === 0) {\n <span class=\"where-text flex-1\">Where</span>\n }\n @for (innerField of subField.fieldGroup; track innerField.key || $index) {\n @if(!!innerField.props['hasDynamicType']){\n @switch (getFilterType(innerField)) {\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\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 }@else{\n <formly-field\n class=\"col\"\n [field]=\"innerField\"\n >\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\"\n (onClickEvent)=\"add()\"\n [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n type=\"outline\" 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\"\n (onClickEvent)=\"reset()\"\n 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:hover{transform:translate(2px)}.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}.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"] }]
190
184
  }] });
191
- //# 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, 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 { MatSelectModule } from '@angular/material/select';\nimport { MatOptionModule } from '@angular/material/core';\nimport { MatIconModule } from '@angular/material/icon';\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  imports: [\n    CommonModule,\n    ReactiveFormsModule,\n    FormlyModule,\n    ButtonComponent,\n    IconComponent,\n    MatFormFieldModule,\n    MatSelectModule,\n    MatOptionModule,\n    MatIconModule    \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('condition')?.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 condition\n    \n    const ruleGroup = this.formControl.at(index) as FormGroup;\n    ruleGroup.get('condition')?.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  // 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  copied: boolean = false;\n  copy(index: number): void {\n    const rowModel = this.formControl.at(index)?.value;\n    if (!rowModel) return;\n  \n    // Deep-clone model\n    const cloned = JSON.parse(JSON.stringify(rowModel));\n  \n    // Insert row after the current one\n    this.add(index + 1, cloned);\n  \n    // 👇 Force Formly to refresh the dynamic field\n    const newRow = this.formControl.at(index + 1);\n    const attr = newRow?.get('attribute')?.value;\n  \n    if (attr) {\n      const attrCtrl = newRow?.get('attribute');\n      attrCtrl.setValue(attrCtrl.value);\n      // Re-trigger field type rebuild for filterValue\n      const filterValueField = this.field.fieldGroup[index + 1].fieldGroup\n        .find(f => f.key === 'filterValue');\n  \n        if (filterValueField) {\n          const typeMap = {\n            SELECT: 'sa-ng-select',\n            STRING: 'sa-input',\n            NUMBER: 'sa-input',\n            DATE: 'sa-date-picker',\n            CURRENCY: 'sa-currency-input',\n            BOOLEAN: 'sa-ng-select',\n          };\n        \n          const newType = typeMap[attr.filterType] || 'sa-input';\n          filterValueField.type = newType;\n          filterValueField.props = {\n            ...this.createFieldProps(attr.filterType),\n            hasDynamicType: true,\n            disabled: false,  // initial\n            attribute: attr,\n          };\n        \n          // 🔄 Notify Formly of type change\n          filterValueField.options?.fieldChanges?.next({\n            type: 'type',\n            field: filterValueField,\n            value: newType,\n          });\n        \n        }\n        \n    }\n  }\n  createFieldProps(type: string) {\n    switch (type) {\n      case 'SELECT':\n        return {\n          placeholder: 'Select',\n          multiple: false,\n          bindLabel: 'label',\n          bindValue: 'value',\n          appearance: 'outline',\n          isClearable: true,\n          isSearchable: false,\n          closeOnselect: true,\n          required: true,\n          disabled: false,\n        };\n      case 'STRING':\n        return {\n          placeholder: 'Enter value',\n          appearance: 'outline',\n          type: 'text'\n        };\n      case 'NUMBER':\n        return {\n          placeholder: 'Enter value',\n          appearance: 'outline',\n          type: 'number'\n        };\n      case 'DATE':\n        return {\n          placeholder: 'Select date',\n          appearance: 'outline',\n          type: 'date'\n        };\n      case 'CURRENCY':\n        return {\n          placeholder: 'Enter value',\n          appearance: 'outline',\n        };\n      case 'BOOLEAN':\n        return {\n          placeholder: 'Enter value',\n          appearance: 'outline',\n          type: 'boolean',\n          options: [\n            { value: true, label: 'True' },\n            { value: false, label: 'False' },\n          ],\n          bindLabel: 'label',\n          bindValue: 'value',\n          disabled: false,\n        };\n      default:\n        return {\n          placeholder: 'Enter value',\n          appearance: 'outline',\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 $index) {\n      <div class=\"row align-items-baseline\">\n        @if ($index === 0) {\n          <span class=\"where-text flex-1\">Where</span>\n        }\n        @for (subField of subField.fieldGroup; track $index) {\n          @if(!!subField.props['hasDynamicType']){\n            @switch (getFilterType(subField)) {\n              @case ('STRING') {\n                <formly-field [field]=\"subField\"></formly-field>\n                }\n                @case ('NUMBER') {\n                <formly-field [field]=\"subField\"></formly-field>\n                }\n                @case ('BOOLEAN') {\n                  <formly-field [field]=\"subField\"></formly-field>\n                  }\n                  @case ('CURRENCY') {\n                    <formly-field [field]=\"subField\"></formly-field>\n                  }\n                  @case ('DATE') {\n\n                  }\n                  @case ('SELECT') {\n                  <formly-field [field]=\"createField(subField)\"></formly-field>\n                  }\n                  @default {\n                  <formly-field [field]=\"createField(subField)\"></formly-field>\n                  }\n            }\n          }@else{\n            <formly-field\n            class=\"col\"\n            [field]=\"subField\"\n          >\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\"\n      (onClickEvent)=\"add()\"\n      [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n      type=\"outline\" 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\"\n      (onClickEvent)=\"reset()\"\n      type=\"primary\" size=\"medium\">\n      </sa-button>\n      }\n    </div>    \n    </div>\n</div>"]}
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":"AAAA,OAAO,EAAE,SAAS,EAAU,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,cAAc,EAAqB,MAAM,kBAAkB,CAAC;AACrE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAChD,OAAO,EAAa,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAChE,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AACnC,OAAO,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAC7D,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAC3D,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,OAAO,EAAS,KAAK,EAAE,UAAU,EAAE,OAAO,EAAkB,MAAM,qBAAqB,CAAC;;;AAmDjG,MAAM,OAAO,2BAA4B,SAAQ,cAAc;IAE7D,QAAQ;IAGR,CAAC;IAED,eAAe,CAAC,KAAa;QACzB,IAAI,KAAK,KAAK,CAAC;YAAE,OAAO,KAAK,CAAC;QAE9B,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,KAAK,CAAc,CAAC;QAC1D,OAAO,SAAS,CAAC,GAAG,CAAC,iBAAiB,CAAC,EAAE,KAAK,IAAI,KAAK,CAAC;IAC1D,CAAC;IAED;;;;KAIC;IACH,eAAe,CAAC,KAAa,EAAE,SAAuB;QACpD,IAAI,KAAK,KAAK,CAAC;YAAE,OAAO,CAAC,4CAA4C;QAErE,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,KAAK,CAAc,CAAC;QAC1D,SAAS,CAAC,GAAG,CAAC,iBAAiB,CAAC,EAAE,QAAQ,CAAC,SAAS,CAAC,CAAC;IACxD,CAAC;IAED,QAAQ;IAER,CAAC;IAED,WAAW,CAAC,KAAwB;QAClC,IAAG,KAAK,CAAC,IAAI,KAAK,cAAc,IAAI,KAAK,CAAC,IAAI,KAAK,EAAE,EAAE,CAAC;YACtD,KAAK,CAAC,IAAI,GAAG,UAAU,CAAC;YACxB,KAAK,CAAC,KAAK,CAAC,IAAI,GAAG,MAAM,CAAC;YAC1B,KAAK,CAAC,KAAK,CAAC,WAAW,GAAG,cAAc,CAAC;QAC3C,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,aAAa,CAAC,QAA2B;QACvC,MAAM,WAAW,GAAG,GAAG,CAAC,QAAQ,EAAE,0BAA0B,CAAC,CAAC;QAC9D,IAAI,CAAC,WAAW;YAAE,OAAO,SAAS,CAAC;QAEnC,4CAA4C;QAC5C,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;YACpC,MAAM,GAAG,GAAG,WAAW,CAAC,GAAG,CAAC,CAAC;YAC7B,IAAI,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,YAAY,IAAI,GAAG,EAAE,CAAC;gBAC1D,OAAO,GAAG,CAAC,UAAU,CAAC;YACxB,CAAC;QACH,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,kBAAkB;IAClB,8BAA8B;IAC9B,IAAI;IAEJ;;;OAGG;IACH,IAAI,CAAC,KAAa;QAChB,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,KAAK,CAAC;QACnD,IAAI,CAAC,QAAQ;YAAE,OAAO;QAEtB,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC;QAEzD,yEAAyE;QACzE,IAAI,WAAW,CAAC,WAAW,KAAK,SAAS,EAAE,CAAC;YAC1C,WAAW,CAAC,YAAY,GAAG,WAAW,CAAC,WAAW,CAAC;QACrD,CAAC;QAED,2FAA2F;QAC3F,wDAAwD;QACxD,IAAI,KAAK,KAAK,CAAC,EAAE,CAAC;YAChB,WAAW,CAAC,eAAe,GAAG,KAAK,CAAC;QACtC,CAAC;QAED,IAAI,CAAC,GAAG,CAAC,KAAK,GAAG,CAAC,EAAE,WAAW,CAAC,CAAC;QAEjC,wEAAwE;QACxE,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,KAAK,GAAG,CAAC,CAAc,CAAC;YAC/D,MAAM,QAAQ,GAAG,UAAU,EAAE,GAAG,CAAC,WAAW,CAAC,CAAC;YAC9C,MAAM,eAAe,GAAG,UAAU,EAAE,GAAG,CAAC,aAAa,CAAC,CAAC;YAEvD,IAAI,QAAQ,IAAI,eAAe,IAAI,WAAW,CAAC,SAAS,EAAE,CAAC;gBACzD,qEAAqE;gBACrE,MAAM,WAAW,GAAG,WAAW,CAAC,YAAY,CAAC;gBAE7C,0DAA0D;gBAC1D,QAAQ,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;gBAE9C,oDAAoD;gBACpD,UAAU,CAAC,GAAG,EAAE;oBACd,QAAQ,CAAC,QAAQ,CAAC,WAAW,CAAC,SAAS,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;oBAE9D,+DAA+D;oBAC/D,UAAU,CAAC,GAAG,EAAE;wBACd,IAAI,WAAW,KAAK,SAAS,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC;4BAChE,wDAAwD;4BACxD,UAAU,CAAC,UAAU,CAAC,EAAE,YAAY,EAAE,WAAW,EAAE,CAAC,CAAC;4BACrD,eAAe,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;wBACxC,CAAC;oBACH,CAAC,EAAE,EAAE,CAAC,CAAC;gBACT,CAAC,EAAE,EAAE,CAAC,CAAC;YACT,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK;QACH,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ;QACvE,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,2BAA2B;IACzC,CAAC;8GAjHU,2BAA2B;kGAA3B,2BAA2B,sEA3C3B,EAAE,iDCpBf,w9EA+DM,8hEDXF,YAAY,8BACZ,mBAAmB,8BACnB,YAAY,yHACZ,eAAe,wRACf,aAAa,oMACb,kBAAkB,8BAClB,eAAe,8BACf,eAAe,8BACf,aAAa,iBAvCH;YACV,OAAO,CAAC,cAAc,EAAE;gBACtB,UAAU,CAAC,QAAQ,EAAE;oBACnB,KAAK,CAAC;wBACJ,OAAO,EAAE,CAAC;wBACV,SAAS,EAAE,+BAA+B;wBAC1C,MAAM,EAAE,KAAK;wBACb,QAAQ,EAAE,QAAQ;qBACnB,CAAC;oBACF,OAAO,CAAC,sCAAsC,EAAE,KAAK,CAAC;wBACpD,OAAO,EAAE,CAAC;wBACV,SAAS,EAAE,wBAAwB;wBACnC,MAAM,EAAE,GAAG;qBACZ,CAAC,CAAC;iBACJ,CAAC;gBACF,UAAU,CAAC,QAAQ,EAAE;oBACnB,KAAK,CAAC;wBACJ,OAAO,EAAE,CAAC;wBACV,SAAS,EAAE,wBAAwB;wBACnC,MAAM,EAAE,GAAG;qBACZ,CAAC;oBACF,OAAO,CAAC,sCAAsC,EAAE,KAAK,CAAC;wBACpD,OAAO,EAAE,CAAC;wBACV,SAAS,EAAE,+BAA+B;wBAC1C,MAAM,EAAE,KAAK;wBACb,QAAQ,EAAE,QAAQ;qBACnB,CAAC,CAAC;iBACJ,CAAC;aACH,CAAC;SACH;;2FAaU,2BAA2B;kBAhDvC,SAAS;+BACE,yBAAyB,cAGvB,IAAI,aACL,EAAE,cACD;wBACV,OAAO,CAAC,cAAc,EAAE;4BACtB,UAAU,CAAC,QAAQ,EAAE;gCACnB,KAAK,CAAC;oCACJ,OAAO,EAAE,CAAC;oCACV,SAAS,EAAE,+BAA+B;oCAC1C,MAAM,EAAE,KAAK;oCACb,QAAQ,EAAE,QAAQ;iCACnB,CAAC;gCACF,OAAO,CAAC,sCAAsC,EAAE,KAAK,CAAC;oCACpD,OAAO,EAAE,CAAC;oCACV,SAAS,EAAE,wBAAwB;oCACnC,MAAM,EAAE,GAAG;iCACZ,CAAC,CAAC;6BACJ,CAAC;4BACF,UAAU,CAAC,QAAQ,EAAE;gCACnB,KAAK,CAAC;oCACJ,OAAO,EAAE,CAAC;oCACV,SAAS,EAAE,wBAAwB;oCACnC,MAAM,EAAE,GAAG;iCACZ,CAAC;gCACF,OAAO,CAAC,sCAAsC,EAAE,KAAK,CAAC;oCACpD,OAAO,EAAE,CAAC;oCACV,SAAS,EAAE,+BAA+B;oCAC1C,MAAM,EAAE,KAAK;oCACb,QAAQ,EAAE,QAAQ;iCACnB,CAAC,CAAC;6BACJ,CAAC;yBACH,CAAC;qBACH,WACQ;wBACP,YAAY;wBACZ,mBAAmB;wBACnB,YAAY;wBACZ,eAAe;wBACf,aAAa;wBACb,kBAAkB;wBAClB,eAAe;wBACf,eAAe;wBACf,aAAa;qBACd","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, 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 { MatSelectModule } from '@angular/material/select';\nimport { MatOptionModule } from '@angular/material/core';\nimport { MatIconModule } from '@angular/material/icon';\nimport { trigger, state, style, transition, animate, query, stagger } from '@angular/animations';\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    MatSelectModule,\n    MatOptionModule,\n    MatIconModule    \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  // 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  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    this.add(index + 1, clonedModel);\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 filterValueCtrl = newRowForm?.get('filterValue');\n      \n      if (attrCtrl && filterValueCtrl && clonedModel.attribute) {\n        // Store the copied value in the form model before triggering rebuild\n        const copiedValue = clonedModel._copiedValue;\n        \n        // Temporarily clear the attribute to force a change event\n        attrCtrl.setValue(null, { emitEvent: false });\n        \n        // Then restore it to trigger the field type rebuild\n        setTimeout(() => {\n          attrCtrl.setValue(clonedModel.attribute, { emitEvent: true });\n          \n          // Ensure the copied value is still available after the rebuild\n          setTimeout(() => {\n            if (copiedValue !== undefined && !newRowForm.value._copiedValue) {\n              // Re-add the copied value if it was lost during rebuild\n              newRowForm.patchValue({ _copiedValue: copiedValue });\n              filterValueCtrl.setValue(copiedValue);\n            }\n          }, 50);\n        }, 10);\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        @if ($index === 0) {\n          <span class=\"where-text flex-1\">Where</span>\n        }\n        @for (innerField of subField.fieldGroup; track innerField.key || $index) {\n          @if(!!innerField.props['hasDynamicType']){\n            @switch (getFilterType(innerField)) {\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\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          }@else{\n            <formly-field\n            class=\"col\"\n            [field]=\"innerField\"\n          >\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\"\n      (onClickEvent)=\"add()\"\n      [state]=\"field?.fieldGroup?.length === (props?.['maxRows'] || 5) ? 'disabled' : 'default'\"\n      type=\"outline\" 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\"\n      (onClickEvent)=\"reset()\"\n      type=\"primary\" size=\"medium\">\n      </sa-button>\n      }\n    </div>    \n    </div>\n</div>"]}