@sarasanalytics-com/design-system 0.0.136 → 0.0.138

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (44) hide show
  1. package/esm2022/interfaces/button-interface.mjs +1 -1
  2. package/esm2022/interfaces/chip-interface.mjs +1 -1
  3. package/esm2022/interfaces/form-layout.interface.mjs +2 -0
  4. package/esm2022/interfaces/grid-interface.mjs +1 -1
  5. package/esm2022/lib/avatar/avatar.component.mjs +3 -3
  6. package/esm2022/lib/categories-nav/categories-nav.component.mjs +27 -0
  7. package/esm2022/lib/chips/chips.component.mjs +3 -3
  8. package/esm2022/lib/data-grid/data-grid.component.mjs +135 -0
  9. package/esm2022/lib/dialog/dialog.component.mjs +43 -3
  10. package/esm2022/lib/filter/filter.component.mjs +232 -0
  11. package/esm2022/lib/form-input/form-input.component.mjs +71 -7
  12. package/esm2022/lib/form-select/form-select.component.mjs +187 -5
  13. package/esm2022/lib/grid-cell/grid-cell.component.mjs +173 -25
  14. package/esm2022/lib/menu/menu-list/menu-item.component.mjs +3 -3
  15. package/esm2022/lib/menu/menu.component.mjs +3 -3
  16. package/esm2022/lib/message-banner-v2/message-banner-v2.component.mjs +256 -7
  17. package/esm2022/lib/query-builder/query-builder-demo.component.mjs +134 -0
  18. package/esm2022/lib/query-builder/query-builder.component.mjs +275 -0
  19. package/esm2022/lib/query-builder/query-builder.service.mjs +107 -0
  20. package/esm2022/lib/query-builder-textarea/query-builder-textarea-demo.component.mjs +130 -0
  21. package/esm2022/lib/query-builder-textarea/query-builder-textarea.component.mjs +805 -0
  22. package/esm2022/lib/tool-tip/tool-tip.component.mjs +8 -3
  23. package/esm2022/public-api.mjs +10 -1
  24. package/fesm2022/sarasanalytics-com-design-system.mjs +3042 -587
  25. package/fesm2022/sarasanalytics-com-design-system.mjs.map +1 -1
  26. package/interfaces/button-interface.d.ts +1 -0
  27. package/interfaces/form-layout.interface.d.ts +59 -0
  28. package/interfaces/grid-interface.d.ts +11 -4
  29. package/lib/categories-nav/categories-nav.component.d.ts +15 -0
  30. package/lib/data-grid/data-grid.component.d.ts +33 -0
  31. package/lib/dialog/dialog.component.d.ts +13 -0
  32. package/lib/filter/filter.component.d.ts +83 -0
  33. package/lib/form-input/form-input.component.d.ts +11 -1
  34. package/lib/form-select/form-select.component.d.ts +8 -0
  35. package/lib/grid-cell/grid-cell.component.d.ts +51 -10
  36. package/lib/message-banner-v2/message-banner-v2.component.d.ts +34 -2
  37. package/lib/query-builder/query-builder-demo.component.d.ts +14 -0
  38. package/lib/query-builder/query-builder.component.d.ts +90 -0
  39. package/lib/query-builder/query-builder.service.d.ts +39 -0
  40. package/lib/query-builder-textarea/query-builder-textarea-demo.component.d.ts +9 -0
  41. package/lib/query-builder-textarea/query-builder-textarea.component.d.ts +143 -0
  42. package/lib/tool-tip/tool-tip.component.d.ts +2 -1
  43. package/package.json +1 -1
  44. package/public-api.d.ts +9 -0
@@ -0,0 +1,134 @@
1
+ import { Component } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { QueryBuilderComponent } from './query-builder.component';
4
+ import { MatCardModule } from '@angular/material/card';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "./query-builder.service";
7
+ import * as i2 from "@angular/common";
8
+ import * as i3 from "@angular/material/card";
9
+ export class QueryBuilderDemoComponent {
10
+ constructor(queryBuilderService) {
11
+ this.queryBuilderService = queryBuilderService;
12
+ this.queryConfig = {
13
+ fields: [
14
+ {
15
+ key: 'name',
16
+ label: 'Name',
17
+ type: 'string'
18
+ },
19
+ {
20
+ key: 'age',
21
+ label: 'Age',
22
+ type: 'number',
23
+ operators: ['=', '!=', '>', '<', '>=', '<=', 'between']
24
+ },
25
+ {
26
+ key: 'email',
27
+ label: 'Email',
28
+ type: 'string',
29
+ operators: ['=', '!=', 'contains']
30
+ },
31
+ {
32
+ key: 'birthdate',
33
+ label: 'Birth Date',
34
+ type: 'date'
35
+ },
36
+ {
37
+ key: 'status',
38
+ label: 'Status',
39
+ type: 'select',
40
+ options: [
41
+ { label: 'Active', value: 'active' },
42
+ { label: 'Inactive', value: 'inactive' },
43
+ { label: 'Pending', value: 'pending' }
44
+ ]
45
+ },
46
+ {
47
+ key: 'isVerified',
48
+ label: 'Is Verified',
49
+ type: 'boolean'
50
+ },
51
+ {
52
+ key: 'lifecycle',
53
+ label: 'Lifecycle',
54
+ type: 'select',
55
+ options: [
56
+ { label: 'Active', value: 'active' },
57
+ { label: 'At Risk', value: 'at_risk' },
58
+ { label: 'Churned', value: 'churned' }
59
+ ]
60
+ }
61
+ ],
62
+ operators: [
63
+ { value: '=', label: '=' },
64
+ { value: '!=', label: '!=' },
65
+ { value: '>', label: '>' },
66
+ { value: '<', label: '<' },
67
+ { value: '>=', label: '>=' },
68
+ { value: '<=', label: '<=' },
69
+ { value: 'contains', label: 'contains' },
70
+ { value: 'between', label: 'between' }
71
+ ],
72
+ defaultCondition: 'and'
73
+ };
74
+ this.queryString = '';
75
+ this.queryJson = '';
76
+ this.currentQuery = null;
77
+ }
78
+ onQueryChange(query) {
79
+ this.currentQuery = query;
80
+ this.queryString = this.queryBuilderService.queryToString(query);
81
+ this.queryJson = JSON.stringify(this.queryBuilderService.queryToJson(query), null, 2);
82
+ }
83
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.4", ngImport: i0, type: QueryBuilderDemoComponent, deps: [{ token: i1.QueryBuilderService }], target: i0.ɵɵFactoryTarget.Component }); }
84
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.2.4", type: QueryBuilderDemoComponent, isStandalone: true, selector: "sa-query-builder-demo", ngImport: i0, template: `
85
+ <mat-card>
86
+ <mat-card-header>
87
+ <mat-card-title>Query Builder Demo</mat-card-title>
88
+ </mat-card-header>
89
+ <mat-card-content>
90
+ <sa-query-builder [config]="queryConfig" (queryChange)="onQueryChange($event)"></sa-query-builder>
91
+
92
+ <div class="query-result" *ngIf="queryString">
93
+ <h3>Query Result:</h3>
94
+ <pre>{{ queryString }}</pre>
95
+ <h3>JSON:</h3>
96
+ <pre>{{ queryJson }}</pre>
97
+ <h3>Per-Row Conditions:</h3>
98
+ <ul>
99
+ <li *ngFor="let rule of currentQuery?.rules; let i = index">
100
+ Row {{ i + 1 }}: {{ i === 0 ? 'First row' : rule.condition?.toUpperCase() }}
101
+ </li>
102
+ </ul>
103
+ </div>
104
+ </mat-card-content>
105
+ </mat-card>
106
+ `, isInline: true, styles: [".query-result{margin-top:20px;padding:16px;background-color:#f5f5f5;border-radius:4px}pre{white-space:pre-wrap;word-break:break-all;background-color:#e0e0e0;padding:8px;border-radius:4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: QueryBuilderComponent, selector: "sa-query-builder", inputs: ["config"], outputs: ["queryChange"] }, { kind: "ngmodule", type: MatCardModule }, { kind: "component", type: i3.MatCard, selector: "mat-card", inputs: ["appearance"], exportAs: ["matCard"] }, { kind: "directive", type: i3.MatCardContent, selector: "mat-card-content" }, { kind: "component", type: i3.MatCardHeader, selector: "mat-card-header" }, { kind: "directive", type: i3.MatCardTitle, selector: "mat-card-title, [mat-card-title], [matCardTitle]" }] }); }
107
+ }
108
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImport: i0, type: QueryBuilderDemoComponent, decorators: [{
109
+ type: Component,
110
+ args: [{ selector: 'sa-query-builder-demo', standalone: true, imports: [CommonModule, QueryBuilderComponent, MatCardModule], template: `
111
+ <mat-card>
112
+ <mat-card-header>
113
+ <mat-card-title>Query Builder Demo</mat-card-title>
114
+ </mat-card-header>
115
+ <mat-card-content>
116
+ <sa-query-builder [config]="queryConfig" (queryChange)="onQueryChange($event)"></sa-query-builder>
117
+
118
+ <div class="query-result" *ngIf="queryString">
119
+ <h3>Query Result:</h3>
120
+ <pre>{{ queryString }}</pre>
121
+ <h3>JSON:</h3>
122
+ <pre>{{ queryJson }}</pre>
123
+ <h3>Per-Row Conditions:</h3>
124
+ <ul>
125
+ <li *ngFor="let rule of currentQuery?.rules; let i = index">
126
+ Row {{ i + 1 }}: {{ i === 0 ? 'First row' : rule.condition?.toUpperCase() }}
127
+ </li>
128
+ </ul>
129
+ </div>
130
+ </mat-card-content>
131
+ </mat-card>
132
+ `, styles: [".query-result{margin-top:20px;padding:16px;background-color:#f5f5f5;border-radius:4px}pre{white-space:pre-wrap;word-break:break-all;background-color:#e0e0e0;padding:8px;border-radius:4px}\n"] }]
133
+ }], ctorParameters: () => [{ type: i1.QueryBuilderService }] });
134
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-builder-demo.component.js","sourceRoot":"","sources":["../../../../../projects/component-library/src/lib/query-builder/query-builder-demo.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { QueryBuilderComponent, QueryConfig, QueryGroup } from './query-builder.component';\r\nimport { QueryBuilderService } from './query-builder.service';\r\nimport { MatCardModule } from '@angular/material/card';\r\n\r\n@Component({\r\n  selector: 'sa-query-builder-demo',\r\n  standalone: true,\r\n  imports: [CommonModule, QueryBuilderComponent, MatCardModule],\r\n  template: `\r\n    <mat-card>\r\n      <mat-card-header>\r\n        <mat-card-title>Query Builder Demo</mat-card-title>\r\n      </mat-card-header>\r\n      <mat-card-content>\r\n        <sa-query-builder [config]=\"queryConfig\" (queryChange)=\"onQueryChange($event)\"></sa-query-builder>\r\n        \r\n        <div class=\"query-result\" *ngIf=\"queryString\">\r\n          <h3>Query Result:</h3>\r\n          <pre>{{ queryString }}</pre>\r\n          <h3>JSON:</h3>\r\n          <pre>{{ queryJson }}</pre>\r\n          <h3>Per-Row Conditions:</h3>\r\n          <ul>\r\n            <li *ngFor=\"let rule of currentQuery?.rules; let i = index\">\r\n              Row {{ i + 1 }}: {{ i === 0 ? 'First row' : rule.condition?.toUpperCase() }}\r\n            </li>\r\n          </ul>\r\n        </div>\r\n      </mat-card-content>\r\n    </mat-card>\r\n  `,\r\n  styles: [`\r\n    .query-result {\r\n      margin-top: 20px;\r\n      padding: 16px;\r\n      background-color: #f5f5f5;\r\n      border-radius: 4px;\r\n    }\r\n    \r\n    pre {\r\n      white-space: pre-wrap;\r\n      word-break: break-all;\r\n      background-color: #e0e0e0;\r\n      padding: 8px;\r\n      border-radius: 4px;\r\n    }\r\n  `]\r\n})\r\nexport class QueryBuilderDemoComponent {\r\n  queryConfig: QueryConfig = {\r\n    fields: [\r\n      {\r\n        key: 'name',\r\n        label: 'Name',\r\n        type: 'string'\r\n      },\r\n      {\r\n        key: 'age',\r\n        label: 'Age',\r\n        type: 'number',\r\n        operators: ['=', '!=', '>', '<', '>=', '<=', 'between']\r\n      },\r\n      {\r\n        key: 'email',\r\n        label: 'Email',\r\n        type: 'string',\r\n        operators: ['=', '!=', 'contains']\r\n      },\r\n      {\r\n        key: 'birthdate',\r\n        label: 'Birth Date',\r\n        type: 'date'\r\n      },\r\n      {\r\n        key: 'status',\r\n        label: 'Status',\r\n        type: 'select',\r\n        options: [\r\n          { label: 'Active', value: 'active' },\r\n          { label: 'Inactive', value: 'inactive' },\r\n          { label: 'Pending', value: 'pending' }\r\n        ]\r\n      },\r\n      {\r\n        key: 'isVerified',\r\n        label: 'Is Verified',\r\n        type: 'boolean'\r\n      },\r\n      {\r\n        key: 'lifecycle',\r\n        label: 'Lifecycle',\r\n        type: 'select',\r\n        options: [\r\n          { label: 'Active', value: 'active' },\r\n          { label: 'At Risk', value: 'at_risk' },\r\n          { label: 'Churned', value: 'churned' }\r\n        ]\r\n      }\r\n    ],\r\n    operators: [\r\n      { value: '=', label: '=' },\r\n      { value: '!=', label: '!=' },\r\n      { value: '>', label: '>' },\r\n      { value: '<', label: '<' },\r\n      { value: '>=', label: '>=' },\r\n      { value: '<=', label: '<=' },\r\n      { value: 'contains', label: 'contains' },\r\n      { value: 'between', label: 'between' }\r\n    ],\r\n    defaultCondition: 'and'\r\n  };\r\n\r\n  queryString: string = '';\r\n  queryJson: string = '';\r\n  currentQuery: QueryGroup | null = null;\r\n\r\n  constructor(private queryBuilderService: QueryBuilderService) {}\r\n\r\n  onQueryChange(query: QueryGroup): void {\r\n    this.currentQuery = query;\r\n    this.queryString = this.queryBuilderService.queryToString(query);\r\n    this.queryJson = JSON.stringify(this.queryBuilderService.queryToJson(query), null, 2);\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,275 @@
1
+ import { Component, EventEmitter, Input, Output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ReactiveFormsModule } from '@angular/forms';
4
+ import { FormlyModule } from '@ngx-formly/core';
5
+ import { MatButtonModule } from '@angular/material/button';
6
+ import { MatIconModule } from '@angular/material/icon';
7
+ import { MatSelectModule } from '@angular/material/select';
8
+ import { MatFormFieldModule } from '@angular/material/form-field';
9
+ import { MatInputModule } from '@angular/material/input';
10
+ import { MatTooltipModule } from '@angular/material/tooltip';
11
+ import { MatButtonToggleModule } from '@angular/material/button-toggle';
12
+ import { MatDatepickerModule } from '@angular/material/datepicker';
13
+ import { provideNativeDateAdapter } from '@angular/material/core';
14
+ import * as i0 from "@angular/core";
15
+ import * as i1 from "@angular/forms";
16
+ import * as i2 from "@angular/material/button";
17
+ import * as i3 from "@angular/material/icon";
18
+ import * as i4 from "@angular/material/form-field";
19
+ import * as i5 from "@angular/material/select";
20
+ import * as i6 from "@angular/material/core";
21
+ import * as i7 from "@angular/material/input";
22
+ import * as i8 from "@angular/material/tooltip";
23
+ import * as i9 from "@angular/material/datepicker";
24
+ export class QueryBuilderComponent {
25
+ constructor(fb) {
26
+ this.fb = fb;
27
+ this.config = {
28
+ fields: [],
29
+ operators: [
30
+ { value: '=', label: '=' },
31
+ { value: '!=', label: '!=' },
32
+ { value: '>', label: '>' },
33
+ { value: '<', label: '<' },
34
+ { value: '>=', label: '>=' },
35
+ { value: '<=', label: '<=' },
36
+ { value: 'contains', label: 'contains' },
37
+ { value: 'between', label: 'between' }
38
+ ],
39
+ defaultCondition: 'and'
40
+ };
41
+ this.queryChange = new EventEmitter();
42
+ this.form = this.fb.group({
43
+ condition: ['and'],
44
+ rules: this.fb.array([])
45
+ });
46
+ this.model = {
47
+ condition: 'and',
48
+ rules: []
49
+ };
50
+ }
51
+ ngOnInit() {
52
+ this.initializeForm();
53
+ this.addRule();
54
+ this.form.valueChanges.subscribe(value => {
55
+ this.model = this.transformFormToModel(value);
56
+ this.queryChange.emit(this.model);
57
+ });
58
+ }
59
+ get rulesArray() {
60
+ return this.form.get('rules');
61
+ }
62
+ initializeForm() {
63
+ this.form = this.fb.group({
64
+ condition: [this.config.defaultCondition || 'and'],
65
+ rules: this.fb.array([])
66
+ });
67
+ }
68
+ addRule() {
69
+ if (this.config.fields.length === 0) {
70
+ return;
71
+ }
72
+ const defaultField = this.config.fields[0];
73
+ const defaultOperator = defaultField.defaultOperator || this.getOperatorsForType(defaultField.type)[0]?.value || '=';
74
+ const ruleGroup = this.fb.group({
75
+ field: [defaultField.key],
76
+ operator: [defaultOperator],
77
+ value: [null],
78
+ condition: ['and']
79
+ });
80
+ this.rulesArray.push(ruleGroup);
81
+ }
82
+ removeRule(index) {
83
+ this.rulesArray.removeAt(index);
84
+ }
85
+ getFieldByKey(key) {
86
+ return this.config.fields.find(field => field.key === key);
87
+ }
88
+ getOperatorsForType(type) {
89
+ return this.config.operators.filter(op => !op.types || op.types.includes(type));
90
+ }
91
+ getOperatorsForField(fieldKey) {
92
+ const field = this.getFieldByKey(fieldKey);
93
+ if (!field)
94
+ return [];
95
+ if (field.operators) {
96
+ return this.config.operators.filter(op => field.operators?.includes(op.value));
97
+ }
98
+ return this.getOperatorsForType(field.type);
99
+ }
100
+ getFieldType(fieldKey) {
101
+ const field = this.getFieldByKey(fieldKey);
102
+ return field ? field.type : 'string';
103
+ }
104
+ isOperatorBetween(index) {
105
+ const ruleGroup = this.rulesArray.at(index);
106
+ return ruleGroup?.get('operator')?.value === 'between';
107
+ }
108
+ getFieldOptions(fieldKey) {
109
+ const field = this.getFieldByKey(fieldKey);
110
+ return field?.options || [];
111
+ }
112
+ onFieldChange(index, fieldKey) {
113
+ const ruleGroup = this.rulesArray.at(index);
114
+ const field = this.getFieldByKey(fieldKey);
115
+ if (field) {
116
+ const operators = this.getOperatorsForField(fieldKey);
117
+ const defaultOperator = field.defaultOperator || (operators.length > 0 ? operators[0].value : '=');
118
+ ruleGroup.get('operator')?.setValue(defaultOperator);
119
+ ruleGroup.get('value')?.setValue(null);
120
+ // Reset range values if they exist
121
+ if (ruleGroup.get('valueStart')) {
122
+ ruleGroup.get('valueStart')?.setValue(null);
123
+ }
124
+ if (ruleGroup.get('valueEnd')) {
125
+ ruleGroup.get('valueEnd')?.setValue(null);
126
+ }
127
+ // Add range form controls if needed
128
+ this.updateFormControlsForOperator(index, defaultOperator, field.type);
129
+ }
130
+ }
131
+ onOperatorChange(index, operator, fieldKey) {
132
+ const fieldType = this.getFieldType(fieldKey);
133
+ this.updateFormControlsForOperator(index, operator, fieldType);
134
+ // Reset values when operator changes
135
+ const ruleGroup = this.rulesArray.at(index);
136
+ ruleGroup.get('value')?.setValue(null);
137
+ if (ruleGroup.get('valueStart')) {
138
+ ruleGroup.get('valueStart')?.setValue(null);
139
+ }
140
+ if (ruleGroup.get('valueEnd')) {
141
+ ruleGroup.get('valueEnd')?.setValue(null);
142
+ }
143
+ }
144
+ /**
145
+ * Gets the minimum allowed date for the end date picker based on the selected start date
146
+ * @param index The index of the rule row
147
+ * @returns The minimum date for the end date picker (the start date or today if no start date)
148
+ */
149
+ getMinEndDate(index) {
150
+ const ruleGroup = this.rulesArray.at(index);
151
+ const startDate = ruleGroup.get('valueStart')?.value;
152
+ // If start date is selected, use it as minimum for end date
153
+ // Otherwise use today's date
154
+ return startDate ? new Date(startDate) : new Date();
155
+ }
156
+ /**
157
+ * Handles changes to the start date
158
+ * @param index The index of the rule row
159
+ * @param event The date change event
160
+ */
161
+ onStartDateChange(index, event) {
162
+ const ruleGroup = this.rulesArray.at(index);
163
+ const endDate = ruleGroup.get('valueEnd')?.value;
164
+ const startDate = event.value;
165
+ // If end date is before start date, reset it
166
+ if (endDate && startDate && new Date(endDate) < new Date(startDate)) {
167
+ ruleGroup.get('valueEnd')?.setValue(null);
168
+ }
169
+ }
170
+ updateFormControlsForOperator(index, operator, fieldType) {
171
+ const ruleGroup = this.rulesArray.at(index);
172
+ // Handle date range for 'between' operator
173
+ if (operator === 'between' && fieldType === 'date') {
174
+ // Add start and end date controls if they don't exist
175
+ if (!ruleGroup.contains('valueStart')) {
176
+ ruleGroup.addControl('valueStart', this.fb.control(null));
177
+ }
178
+ if (!ruleGroup.contains('valueEnd')) {
179
+ ruleGroup.addControl('valueEnd', this.fb.control(null));
180
+ }
181
+ }
182
+ else {
183
+ // Remove range controls if they exist and aren't needed
184
+ if (ruleGroup.contains('valueStart')) {
185
+ ruleGroup.removeControl('valueStart');
186
+ }
187
+ if (ruleGroup.contains('valueEnd')) {
188
+ ruleGroup.removeControl('valueEnd');
189
+ }
190
+ }
191
+ }
192
+ transformFormToModel(formValue) {
193
+ return {
194
+ condition: formValue.condition,
195
+ rules: formValue.rules.map((rule) => {
196
+ // Handle date range for 'between' operator
197
+ if (rule.operator === 'between' && this.getFieldType(rule.field) === 'date') {
198
+ return {
199
+ field: rule.field,
200
+ operator: rule.operator,
201
+ value: { start: rule.valueStart, end: rule.valueEnd },
202
+ condition: rule.condition
203
+ };
204
+ }
205
+ // Standard case
206
+ return {
207
+ field: rule.field,
208
+ operator: rule.operator,
209
+ value: rule.value,
210
+ condition: rule.condition
211
+ };
212
+ })
213
+ };
214
+ }
215
+ /**
216
+ * Gets the condition for a specific row
217
+ * @param index The index of the row
218
+ * @returns The condition (and/or) for the row
219
+ */
220
+ getRowCondition(index) {
221
+ if (index === 0)
222
+ return 'and'; // First row doesn't have a condition
223
+ const ruleGroup = this.rulesArray.at(index);
224
+ return ruleGroup.get('condition')?.value || 'and';
225
+ }
226
+ /**
227
+ * Sets the condition for a specific row
228
+ * @param index The index of the row
229
+ * @param condition The condition to set (and/or)
230
+ */
231
+ setRowCondition(index, condition) {
232
+ if (index === 0)
233
+ return; // First row doesn't have a condition
234
+ const ruleGroup = this.rulesArray.at(index);
235
+ ruleGroup.get('condition')?.setValue(condition);
236
+ }
237
+ toggleCondition() {
238
+ const currentCondition = this.form.get('condition')?.value;
239
+ const newCondition = currentCondition === 'and' ? 'or' : 'and';
240
+ this.form.get('condition')?.setValue(newCondition);
241
+ }
242
+ getConditionLabel() {
243
+ return this.form.get('condition')?.value === 'and' ? 'AND' : 'OR';
244
+ }
245
+ resetQuery() {
246
+ this.rulesArray.clear();
247
+ this.addRule();
248
+ }
249
+ applyQuery() {
250
+ this.queryChange.emit(this.model);
251
+ }
252
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.4", ngImport: i0, type: QueryBuilderComponent, deps: [{ token: i1.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
253
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.2.4", type: QueryBuilderComponent, isStandalone: true, selector: "sa-query-builder", inputs: { config: "config" }, outputs: { queryChange: "queryChange" }, providers: [provideNativeDateAdapter()], ngImport: i0, template: "<div class=\"sa-query-builder\">\r\n <form [formGroup]=\"form\">\r\n <div class=\"query-builder-header\">\r\n <div class=\"where-label\">Where</div>\r\n </div>\r\n\r\n <div class=\"query-builder-body\">\r\n <div class=\"rules-container\" formArrayName=\"rules\">\r\n @for (rule of rulesArray.controls; track $index) {\r\n <div class=\"rule-row\" [formGroupName]=\"$index\">\r\n <!-- Condition selector (And/Or) -->\r\n @if ($index > 0) {\r\n <div class=\"condition-selector\">\r\n <mat-form-field appearance=\"fill\">\r\n <mat-select [value]=\"getRowCondition($index)\" (selectionChange)=\"setRowCondition($index, $event.value)\">\r\n <mat-option value=\"and\">And</mat-option>\r\n <mat-option value=\"or\">Or</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n } @else {\r\n <div class=\"condition-placeholder\"></div>\r\n }\r\n \r\n <!-- Field selection -->\r\n <mat-form-field class=\"field-select\">\r\n <mat-label>Select a filter</mat-label>\r\n <mat-select formControlName=\"field\" (selectionChange)=\"onFieldChange($index, $event.value)\">\r\n @for (field of config.fields; track field.key) {\r\n <mat-option [value]=\"field.key\">\r\n {{ field.label }}\r\n </mat-option>\r\n }\r\n </mat-select>\r\n </mat-form-field>\r\n\r\n <!-- Operator selection -->\r\n <mat-form-field class=\"operator-select\">\r\n <mat-select formControlName=\"operator\" (selectionChange)=\"onOperatorChange($index, $event.value, rule.value.field)\">\r\n @for (op of getOperatorsForField(rule.value.field); track op.value) {\r\n <mat-option [value]=\"op.value\">\r\n {{ op.label }}\r\n </mat-option>\r\n }\r\n </mat-select>\r\n </mat-form-field>\r\n\r\n <!-- Value input based on field type -->\r\n @switch (getFieldType(rule.value.field)) {\r\n @case ('string') {\r\n <mat-form-field class=\"value-input\">\r\n <input matInput formControlName=\"value\" placeholder=\"Enter a value...\">\r\n </mat-form-field>\r\n }\r\n @case ('number') {\r\n <mat-form-field class=\"value-input\">\r\n <input matInput type=\"number\" formControlName=\"value\" placeholder=\"Enter a value...\">\r\n </mat-form-field>\r\n }\r\n @case ('date') {\r\n @if (!isOperatorBetween($index)) {\r\n <mat-form-field class=\"value-input\">\r\n <input matInput type=\"date\" [matDatepicker]=\"picker\" formControlName=\"value\" (click)=\"picker.open()\">\r\n <mat-datepicker-toggle matIconSuffix [for]=\"picker\"></mat-datepicker-toggle>\r\n <mat-datepicker #picker></mat-datepicker>\r\n </mat-form-field>\r\n } @else {\r\n <div class=\"date-range-container\">\r\n <mat-form-field class=\"value-input start-date\">\r\n <mat-label>Start date</mat-label>\r\n <input matInput [matDatepicker]=\"startPicker\" formControlName=\"valueStart\" (click)=\"startPicker.open()\" (dateChange)=\"onStartDateChange($index, $event)\">\r\n <mat-datepicker-toggle matIconSuffix [for]=\"startPicker\"></mat-datepicker-toggle>\r\n <mat-datepicker #startPicker></mat-datepicker>\r\n </mat-form-field>\r\n <mat-form-field class=\"value-input end-date\">\r\n <mat-label>End date</mat-label>\r\n <input matInput [matDatepicker]=\"endPicker\" formControlName=\"valueEnd\" (click)=\"endPicker.open()\" [min]=\"getMinEndDate($index)\">\r\n <mat-datepicker-toggle matIconSuffix [for]=\"endPicker\"></mat-datepicker-toggle>\r\n <mat-datepicker #endPicker [startAt]=\"getMinEndDate($index)\"></mat-datepicker>\r\n </mat-form-field>\r\n </div>\r\n }\r\n }\r\n @case ('boolean') {\r\n <mat-form-field class=\"value-input\">\r\n <mat-select formControlName=\"value\">\r\n <mat-option [value]=\"true\">True</mat-option>\r\n <mat-option [value]=\"false\">False</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n }\r\n @case ('select') {\r\n <mat-form-field class=\"value-input\">\r\n <mat-select formControlName=\"value\">\r\n @for (option of getFieldOptions(rule.value.field); track option.value) {\r\n <mat-option [value]=\"option.value\">\r\n {{ option.label }}\r\n </mat-option>\r\n }\r\n </mat-select>\r\n </mat-form-field>\r\n }\r\n @default {\r\n <mat-form-field class=\"value-input\">\r\n <input matInput formControlName=\"value\" placeholder=\"Enter a value...\">\r\n </mat-form-field>\r\n }\r\n }\r\n\r\n <!-- Remove rule button -->\r\n <button mat-icon-button class=\"remove-rule-btn\" \r\n (click)=\"removeRule($index)\" \r\n [disabled]=\"rulesArray.length === 1\"\r\n matTooltip=\"Remove filter\">\r\n <mat-icon>close</mat-icon>\r\n </button>\r\n </div>\r\n }\r\n </div>\r\n\r\n <!-- Add rule button -->\r\n <div class=\"add-rule-container\">\r\n <button mat-button color=\"primary\" class=\"add-rule-btn\" (click)=\"addRule()\">\r\n <mat-icon>add</mat-icon> Add a filter\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <div class=\"query-builder-footer\">\r\n <button mat-button class=\"reset-btn\" (click)=\"resetQuery()\">Reset</button>\r\n <button mat-raised-button color=\"primary\" class=\"apply-btn\" (click)=\"applyQuery()\">Apply filters</button>\r\n </div>\r\n </form>\r\n</div>\r\n", styles: [".sa-query-builder{display:flex;flex-direction:column;width:100%;background-color:#fff;border-radius:8px;padding:16px;box-sizing:border-box}.sa-query-builder .query-builder-header{display:flex;align-items:center;margin-bottom:16px;height:36px;padding-left:128px}.sa-query-builder .query-builder-header .where-label{font-size:16px;font-weight:500;color:#000000de;margin-right:16px;line-height:36px}.sa-query-builder .query-builder-body .rules-container{display:flex;flex-direction:column;gap:12px;margin-bottom:16px}.sa-query-builder .query-builder-body .rules-container .rule-row{display:flex;align-items:center;gap:12px;width:100%}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-selector{width:120px;min-width:120px;margin-right:8px}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-selector .mat-form-field{width:100%}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-selector ::ng-deep .mat-form-field-wrapper{padding-bottom:0;margin:0}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-selector ::ng-deep .mat-form-field-infix{padding:.5em 0;border-top:0}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-placeholder{width:120px;min-width:120px;margin-right:8px}.sa-query-builder .query-builder-body .rules-container .rule-row .field-select{flex:2;min-width:180px}.sa-query-builder .query-builder-body .rules-container .rule-row .operator-select{flex:1;min-width:100px}.sa-query-builder .query-builder-body .rules-container .rule-row .value-input{flex:2;min-width:180px}.sa-query-builder .query-builder-body .rules-container .rule-row .date-range-container{display:flex;flex:2;gap:8px}.sa-query-builder .query-builder-body .rules-container .rule-row .date-range-container .start-date,.sa-query-builder .query-builder-body .rules-container .rule-row .date-range-container .end-date{flex:1;min-width:140px}.sa-query-builder .query-builder-body .rules-container .rule-row .remove-rule-btn{color:#0000008a}.sa-query-builder .query-builder-body .add-rule-container{margin-bottom:16px}.sa-query-builder .query-builder-body .add-rule-container .add-rule-btn{display:flex;align-items:center;gap:4px;color:#673ab7;font-weight:500}.sa-query-builder .query-builder-footer{display:flex;justify-content:flex-end;gap:12px;margin-top:16px}.sa-query-builder .query-builder-footer .reset-btn{color:#0000008a}.sa-query-builder .query-builder-footer .apply-btn{background-color:#673ab7;color:#fff}::ng-deep .mat-mdc-form-field-subscript-wrapper{display:none}::ng-deep .mat-mdc-form-field-infix{min-height:40px}::ng-deep .mat-mdc-select-panel{max-height:300px}::ng-deep .mat-mdc-option{font-size:14px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { 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.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i1.FormGroupName, selector: "[formGroupName]", inputs: ["formGroupName"] }, { kind: "directive", type: i1.FormArrayName, selector: "[formArrayName]", inputs: ["formArrayName"] }, { kind: "ngmodule", type: FormlyModule }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i2.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i4.MatLabel, selector: "mat-label" }, { kind: "directive", type: i4.MatSuffix, selector: "[matSuffix], [matIconSuffix], [matTextSuffix]", inputs: ["matTextSuffix"] }, { kind: "component", type: i5.MatSelect, selector: "mat-select", inputs: ["aria-describedby", "panelClass", "disabled", "disableRipple", "tabIndex", "hideSingleSelectionIndicator", "placeholder", "required", "multiple", "disableOptionCentering", "compareWith", "value", "aria-label", "aria-labelledby", "errorStateMatcher", "typeaheadDebounceInterval", "sortComparator", "id", "panelWidth"], outputs: ["openedChange", "opened", "closed", "selectionChange", "valueChange"], exportAs: ["matSelect"] }, { kind: "component", type: i6.MatOption, selector: "mat-option", inputs: ["value", "id", "disabled"], outputs: ["onSelectionChange"], exportAs: ["matOption"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i7.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: MatTooltipModule }, { kind: "directive", type: i8.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatButtonToggleModule }, { kind: "ngmodule", type: MatDatepickerModule }, { kind: "component", type: i9.MatDatepicker, selector: "mat-datepicker", exportAs: ["matDatepicker"] }, { kind: "directive", type: i9.MatDatepickerInput, selector: "input[matDatepicker]", inputs: ["matDatepicker", "min", "max", "matDatepickerFilter"], exportAs: ["matDatepickerInput"] }, { kind: "component", type: i9.MatDatepickerToggle, selector: "mat-datepicker-toggle", inputs: ["for", "tabIndex", "aria-label", "disabled", "disableRipple"], exportAs: ["matDatepickerToggle"] }] }); }
254
+ }
255
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImport: i0, type: QueryBuilderComponent, decorators: [{
256
+ type: Component,
257
+ args: [{ selector: 'sa-query-builder', standalone: true, providers: [provideNativeDateAdapter()], imports: [
258
+ CommonModule,
259
+ ReactiveFormsModule,
260
+ FormlyModule,
261
+ MatButtonModule,
262
+ MatIconModule,
263
+ MatSelectModule,
264
+ MatFormFieldModule,
265
+ MatInputModule,
266
+ MatTooltipModule,
267
+ MatButtonToggleModule,
268
+ MatDatepickerModule
269
+ ], template: "<div class=\"sa-query-builder\">\r\n <form [formGroup]=\"form\">\r\n <div class=\"query-builder-header\">\r\n <div class=\"where-label\">Where</div>\r\n </div>\r\n\r\n <div class=\"query-builder-body\">\r\n <div class=\"rules-container\" formArrayName=\"rules\">\r\n @for (rule of rulesArray.controls; track $index) {\r\n <div class=\"rule-row\" [formGroupName]=\"$index\">\r\n <!-- Condition selector (And/Or) -->\r\n @if ($index > 0) {\r\n <div class=\"condition-selector\">\r\n <mat-form-field appearance=\"fill\">\r\n <mat-select [value]=\"getRowCondition($index)\" (selectionChange)=\"setRowCondition($index, $event.value)\">\r\n <mat-option value=\"and\">And</mat-option>\r\n <mat-option value=\"or\">Or</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n </div>\r\n } @else {\r\n <div class=\"condition-placeholder\"></div>\r\n }\r\n \r\n <!-- Field selection -->\r\n <mat-form-field class=\"field-select\">\r\n <mat-label>Select a filter</mat-label>\r\n <mat-select formControlName=\"field\" (selectionChange)=\"onFieldChange($index, $event.value)\">\r\n @for (field of config.fields; track field.key) {\r\n <mat-option [value]=\"field.key\">\r\n {{ field.label }}\r\n </mat-option>\r\n }\r\n </mat-select>\r\n </mat-form-field>\r\n\r\n <!-- Operator selection -->\r\n <mat-form-field class=\"operator-select\">\r\n <mat-select formControlName=\"operator\" (selectionChange)=\"onOperatorChange($index, $event.value, rule.value.field)\">\r\n @for (op of getOperatorsForField(rule.value.field); track op.value) {\r\n <mat-option [value]=\"op.value\">\r\n {{ op.label }}\r\n </mat-option>\r\n }\r\n </mat-select>\r\n </mat-form-field>\r\n\r\n <!-- Value input based on field type -->\r\n @switch (getFieldType(rule.value.field)) {\r\n @case ('string') {\r\n <mat-form-field class=\"value-input\">\r\n <input matInput formControlName=\"value\" placeholder=\"Enter a value...\">\r\n </mat-form-field>\r\n }\r\n @case ('number') {\r\n <mat-form-field class=\"value-input\">\r\n <input matInput type=\"number\" formControlName=\"value\" placeholder=\"Enter a value...\">\r\n </mat-form-field>\r\n }\r\n @case ('date') {\r\n @if (!isOperatorBetween($index)) {\r\n <mat-form-field class=\"value-input\">\r\n <input matInput type=\"date\" [matDatepicker]=\"picker\" formControlName=\"value\" (click)=\"picker.open()\">\r\n <mat-datepicker-toggle matIconSuffix [for]=\"picker\"></mat-datepicker-toggle>\r\n <mat-datepicker #picker></mat-datepicker>\r\n </mat-form-field>\r\n } @else {\r\n <div class=\"date-range-container\">\r\n <mat-form-field class=\"value-input start-date\">\r\n <mat-label>Start date</mat-label>\r\n <input matInput [matDatepicker]=\"startPicker\" formControlName=\"valueStart\" (click)=\"startPicker.open()\" (dateChange)=\"onStartDateChange($index, $event)\">\r\n <mat-datepicker-toggle matIconSuffix [for]=\"startPicker\"></mat-datepicker-toggle>\r\n <mat-datepicker #startPicker></mat-datepicker>\r\n </mat-form-field>\r\n <mat-form-field class=\"value-input end-date\">\r\n <mat-label>End date</mat-label>\r\n <input matInput [matDatepicker]=\"endPicker\" formControlName=\"valueEnd\" (click)=\"endPicker.open()\" [min]=\"getMinEndDate($index)\">\r\n <mat-datepicker-toggle matIconSuffix [for]=\"endPicker\"></mat-datepicker-toggle>\r\n <mat-datepicker #endPicker [startAt]=\"getMinEndDate($index)\"></mat-datepicker>\r\n </mat-form-field>\r\n </div>\r\n }\r\n }\r\n @case ('boolean') {\r\n <mat-form-field class=\"value-input\">\r\n <mat-select formControlName=\"value\">\r\n <mat-option [value]=\"true\">True</mat-option>\r\n <mat-option [value]=\"false\">False</mat-option>\r\n </mat-select>\r\n </mat-form-field>\r\n }\r\n @case ('select') {\r\n <mat-form-field class=\"value-input\">\r\n <mat-select formControlName=\"value\">\r\n @for (option of getFieldOptions(rule.value.field); track option.value) {\r\n <mat-option [value]=\"option.value\">\r\n {{ option.label }}\r\n </mat-option>\r\n }\r\n </mat-select>\r\n </mat-form-field>\r\n }\r\n @default {\r\n <mat-form-field class=\"value-input\">\r\n <input matInput formControlName=\"value\" placeholder=\"Enter a value...\">\r\n </mat-form-field>\r\n }\r\n }\r\n\r\n <!-- Remove rule button -->\r\n <button mat-icon-button class=\"remove-rule-btn\" \r\n (click)=\"removeRule($index)\" \r\n [disabled]=\"rulesArray.length === 1\"\r\n matTooltip=\"Remove filter\">\r\n <mat-icon>close</mat-icon>\r\n </button>\r\n </div>\r\n }\r\n </div>\r\n\r\n <!-- Add rule button -->\r\n <div class=\"add-rule-container\">\r\n <button mat-button color=\"primary\" class=\"add-rule-btn\" (click)=\"addRule()\">\r\n <mat-icon>add</mat-icon> Add a filter\r\n </button>\r\n </div>\r\n </div>\r\n\r\n <div class=\"query-builder-footer\">\r\n <button mat-button class=\"reset-btn\" (click)=\"resetQuery()\">Reset</button>\r\n <button mat-raised-button color=\"primary\" class=\"apply-btn\" (click)=\"applyQuery()\">Apply filters</button>\r\n </div>\r\n </form>\r\n</div>\r\n", styles: [".sa-query-builder{display:flex;flex-direction:column;width:100%;background-color:#fff;border-radius:8px;padding:16px;box-sizing:border-box}.sa-query-builder .query-builder-header{display:flex;align-items:center;margin-bottom:16px;height:36px;padding-left:128px}.sa-query-builder .query-builder-header .where-label{font-size:16px;font-weight:500;color:#000000de;margin-right:16px;line-height:36px}.sa-query-builder .query-builder-body .rules-container{display:flex;flex-direction:column;gap:12px;margin-bottom:16px}.sa-query-builder .query-builder-body .rules-container .rule-row{display:flex;align-items:center;gap:12px;width:100%}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-selector{width:120px;min-width:120px;margin-right:8px}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-selector .mat-form-field{width:100%}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-selector ::ng-deep .mat-form-field-wrapper{padding-bottom:0;margin:0}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-selector ::ng-deep .mat-form-field-infix{padding:.5em 0;border-top:0}.sa-query-builder .query-builder-body .rules-container .rule-row .condition-placeholder{width:120px;min-width:120px;margin-right:8px}.sa-query-builder .query-builder-body .rules-container .rule-row .field-select{flex:2;min-width:180px}.sa-query-builder .query-builder-body .rules-container .rule-row .operator-select{flex:1;min-width:100px}.sa-query-builder .query-builder-body .rules-container .rule-row .value-input{flex:2;min-width:180px}.sa-query-builder .query-builder-body .rules-container .rule-row .date-range-container{display:flex;flex:2;gap:8px}.sa-query-builder .query-builder-body .rules-container .rule-row .date-range-container .start-date,.sa-query-builder .query-builder-body .rules-container .rule-row .date-range-container .end-date{flex:1;min-width:140px}.sa-query-builder .query-builder-body .rules-container .rule-row .remove-rule-btn{color:#0000008a}.sa-query-builder .query-builder-body .add-rule-container{margin-bottom:16px}.sa-query-builder .query-builder-body .add-rule-container .add-rule-btn{display:flex;align-items:center;gap:4px;color:#673ab7;font-weight:500}.sa-query-builder .query-builder-footer{display:flex;justify-content:flex-end;gap:12px;margin-top:16px}.sa-query-builder .query-builder-footer .reset-btn{color:#0000008a}.sa-query-builder .query-builder-footer .apply-btn{background-color:#673ab7;color:#fff}::ng-deep .mat-mdc-form-field-subscript-wrapper{display:none}::ng-deep .mat-mdc-form-field-infix{min-height:40px}::ng-deep .mat-mdc-select-panel{max-height:300px}::ng-deep .mat-mdc-option{font-size:14px}\n"] }]
270
+ }], ctorParameters: () => [{ type: i1.FormBuilder }], propDecorators: { config: [{
271
+ type: Input
272
+ }], queryChange: [{
273
+ type: Output
274
+ }] } });
275
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query-builder.component.js","sourceRoot":"","sources":["../../../../../projects/component-library/src/lib/query-builder/query-builder.component.ts","../../../../../projects/component-library/src/lib/query-builder/query-builder.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormArray, FormBuilder, FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';\r\nimport { FormlyFieldConfig, FormlyModule } from '@ngx-formly/core';\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\nimport { MatButtonToggleModule } from '@angular/material/button-toggle';\r\nimport {MatDatepickerModule} from '@angular/material/datepicker';\r\nimport { provideNativeDateAdapter } from '@angular/material/core';\r\n\r\n\r\nexport interface QueryField {\r\n  key: string;\r\n  label: string;\r\n  type: string;\r\n  options?: { label: string; value: any }[];\r\n  operators?: string[];\r\n  defaultOperator?: string;\r\n}\r\n\r\nexport interface QueryOperator {\r\n  value: string;\r\n  label: string;\r\n  types?: string[];\r\n}\r\n\r\nexport interface QueryConfig {\r\n  fields: QueryField[];\r\n  operators: QueryOperator[];\r\n  defaultCondition?: 'and' | 'or';\r\n}\r\n\r\nexport interface QueryRule {\r\n  field: string;\r\n  operator: string;\r\n  value: any;\r\n  condition?: 'and' | 'or';\r\n}\r\n\r\nexport interface QueryGroup {\r\n  condition: 'and' | 'or';\r\n  rules: (QueryRule | QueryGroup)[];\r\n}\r\n\r\n@Component({\r\n  selector: 'sa-query-builder',\r\n  templateUrl: './query-builder.component.html',\r\n  styleUrls: ['./query-builder.component.scss'],\r\n  standalone: true,\r\n  providers: [provideNativeDateAdapter()],\r\n  imports: [\r\n    CommonModule,\r\n    ReactiveFormsModule,\r\n    FormlyModule,\r\n    MatButtonModule,\r\n    MatIconModule,\r\n    MatSelectModule,\r\n    MatFormFieldModule,\r\n    MatInputModule,\r\n    MatTooltipModule,\r\n    MatButtonToggleModule,\r\n    MatDatepickerModule\r\n  ]\r\n})\r\nexport class QueryBuilderComponent implements OnInit {\r\n  @Input() config: QueryConfig = {\r\n    fields: [],\r\n    operators: [\r\n      { value: '=', label: '=' },\r\n      { value: '!=', label: '!=' },\r\n      { value: '>', label: '>' },\r\n      { value: '<', label: '<' },\r\n      { value: '>=', label: '>=' },\r\n      { value: '<=', label: '<=' },\r\n      { value: 'contains', label: 'contains' },\r\n      { value: 'between', label: 'between' }\r\n    ],\r\n    defaultCondition: 'and'\r\n  };\r\n\r\n  @Output() queryChange = new EventEmitter<QueryGroup>();\r\n\r\n  form: FormGroup;\r\n  model: QueryGroup;\r\n\r\n  constructor(private fb: FormBuilder) {\r\n    this.form = this.fb.group({\r\n      condition: ['and'],\r\n      rules: this.fb.array([])\r\n    });\r\n\r\n    this.model = {\r\n      condition: 'and',\r\n      rules: []\r\n    };\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.initializeForm();\r\n    this.addRule();\r\n\r\n    this.form.valueChanges.subscribe(value => {\r\n      this.model = this.transformFormToModel(value);\r\n      this.queryChange.emit(this.model);\r\n    });\r\n  }\r\n\r\n  get rulesArray(): FormArray {\r\n    return this.form.get('rules') as FormArray;\r\n  }\r\n\r\n  initializeForm(): void {\r\n    this.form = this.fb.group({\r\n      condition: [this.config.defaultCondition || 'and'],\r\n      rules: this.fb.array([])\r\n    });\r\n  }\r\n\r\n  addRule(): void {\r\n    if (this.config.fields.length === 0) {\r\n      return;\r\n    }\r\n\r\n    const defaultField = this.config.fields[0];\r\n    const defaultOperator = defaultField.defaultOperator || this.getOperatorsForType(defaultField.type)[0]?.value || '=';\r\n\r\n    const ruleGroup = this.fb.group({\r\n      field: [defaultField.key],\r\n      operator: [defaultOperator],\r\n      value: [null],\r\n      condition: ['and']\r\n    });\r\n\r\n    this.rulesArray.push(ruleGroup);\r\n  }\r\n\r\n  removeRule(index: number): void {\r\n    this.rulesArray.removeAt(index);\r\n  }\r\n\r\n  getFieldByKey(key: string): QueryField | undefined {\r\n    return this.config.fields.find(field => field.key === key);\r\n  }\r\n\r\n  getOperatorsForType(type: string): QueryOperator[] {\r\n    return this.config.operators.filter(op => !op.types || op.types.includes(type));\r\n  }\r\n\r\n  getOperatorsForField(fieldKey: string): QueryOperator[] {\r\n    const field = this.getFieldByKey(fieldKey);\r\n    if (!field) return [];\r\n\r\n    if (field.operators) {\r\n      return this.config.operators.filter(op => field.operators?.includes(op.value));\r\n    }\r\n\r\n    return this.getOperatorsForType(field.type);\r\n  }\r\n\r\n  getFieldType(fieldKey: string): string {\r\n    const field = this.getFieldByKey(fieldKey);\r\n    return field ? field.type : 'string';\r\n  }\r\n  \r\n  isOperatorBetween(index: number): boolean {\r\n    const ruleGroup = this.rulesArray.at(index) as FormGroup;\r\n    return ruleGroup?.get('operator')?.value === 'between';\r\n  }\r\n\r\n  getFieldOptions(fieldKey: string): { label: string; value: any }[] {\r\n    const field = this.getFieldByKey(fieldKey);\r\n    return field?.options || [];\r\n  }\r\n\r\n  onFieldChange(index: number, fieldKey: string): void {\r\n    const ruleGroup = this.rulesArray.at(index) as FormGroup;\r\n    const field = this.getFieldByKey(fieldKey);\r\n    \r\n    if (field) {\r\n      const operators = this.getOperatorsForField(fieldKey);\r\n      const defaultOperator = field.defaultOperator || (operators.length > 0 ? operators[0].value : '=');\r\n      \r\n      ruleGroup.get('operator')?.setValue(defaultOperator);\r\n      ruleGroup.get('value')?.setValue(null);\r\n      \r\n      // Reset range values if they exist\r\n      if (ruleGroup.get('valueStart')) {\r\n        ruleGroup.get('valueStart')?.setValue(null);\r\n      }\r\n      if (ruleGroup.get('valueEnd')) {\r\n        ruleGroup.get('valueEnd')?.setValue(null);\r\n      }\r\n      \r\n      // Add range form controls if needed\r\n      this.updateFormControlsForOperator(index, defaultOperator, field.type);\r\n    }\r\n  }\r\n\r\n  onOperatorChange(index: number, operator: string, fieldKey: string): void {\r\n    const fieldType = this.getFieldType(fieldKey);\r\n    this.updateFormControlsForOperator(index, operator, fieldType);\r\n    \r\n    // Reset values when operator changes\r\n    const ruleGroup = this.rulesArray.at(index) as FormGroup;\r\n    ruleGroup.get('value')?.setValue(null);\r\n    if (ruleGroup.get('valueStart')) {\r\n      ruleGroup.get('valueStart')?.setValue(null);\r\n    }\r\n    if (ruleGroup.get('valueEnd')) {\r\n      ruleGroup.get('valueEnd')?.setValue(null);\r\n    }\r\n  }\r\n  \r\n  /**\r\n   * Gets the minimum allowed date for the end date picker based on the selected start date\r\n   * @param index The index of the rule row\r\n   * @returns The minimum date for the end date picker (the start date or today if no start date)\r\n   */\r\n  getMinEndDate(index: number): Date | null {\r\n    const ruleGroup = this.rulesArray.at(index) as FormGroup;\r\n    const startDate = ruleGroup.get('valueStart')?.value;\r\n    \r\n    // If start date is selected, use it as minimum for end date\r\n    // Otherwise use today's date\r\n    return startDate ? new Date(startDate) : new Date();\r\n  }\r\n  \r\n  /**\r\n   * Handles changes to the start date\r\n   * @param index The index of the rule row\r\n   * @param event The date change event\r\n   */\r\n  onStartDateChange(index: number, event: any): void {\r\n    const ruleGroup = this.rulesArray.at(index) as FormGroup;\r\n    const endDate = ruleGroup.get('valueEnd')?.value;\r\n    const startDate = event.value;\r\n    \r\n    // If end date is before start date, reset it\r\n    if (endDate && startDate && new Date(endDate) < new Date(startDate)) {\r\n      ruleGroup.get('valueEnd')?.setValue(null);\r\n    }\r\n  }\r\n  \r\n  updateFormControlsForOperator(index: number, operator: string, fieldType: string): void {\r\n    const ruleGroup = this.rulesArray.at(index) as FormGroup;\r\n    \r\n    // Handle date range for 'between' operator\r\n    if (operator === 'between' && fieldType === 'date') {\r\n      // Add start and end date controls if they don't exist\r\n      if (!ruleGroup.contains('valueStart')) {\r\n        ruleGroup.addControl('valueStart', this.fb.control(null));\r\n      }\r\n      if (!ruleGroup.contains('valueEnd')) {\r\n        ruleGroup.addControl('valueEnd', this.fb.control(null));\r\n      }\r\n    } else {\r\n      // Remove range controls if they exist and aren't needed\r\n      if (ruleGroup.contains('valueStart')) {\r\n        ruleGroup.removeControl('valueStart');\r\n      }\r\n      if (ruleGroup.contains('valueEnd')) {\r\n        ruleGroup.removeControl('valueEnd');\r\n      }\r\n    }\r\n  }\r\n  \r\n  transformFormToModel(formValue: any): QueryGroup {\r\n    return {\r\n      condition: formValue.condition,\r\n      rules: formValue.rules.map((rule: any) => {\r\n        // Handle date range for 'between' operator\r\n        if (rule.operator === 'between' && this.getFieldType(rule.field) === 'date') {\r\n          return {\r\n            field: rule.field,\r\n            operator: rule.operator,\r\n            value: { start: rule.valueStart, end: rule.valueEnd },\r\n            condition: rule.condition\r\n          };\r\n        }\r\n        \r\n        // Standard case\r\n        return {\r\n          field: rule.field,\r\n          operator: rule.operator,\r\n          value: rule.value,\r\n          condition: rule.condition\r\n        };\r\n      })\r\n    };\r\n  }\r\n  \r\n  /**\r\n   * Gets the condition for a specific row\r\n   * @param index The index of the row\r\n   * @returns The condition (and/or) for the row\r\n   */\r\n  getRowCondition(index: number): 'and' | 'or' {\r\n    if (index === 0) return 'and'; // First row doesn't have a condition\r\n    \r\n    const ruleGroup = this.rulesArray.at(index) as FormGroup;\r\n    return ruleGroup.get('condition')?.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 condition\r\n    \r\n    const ruleGroup = this.rulesArray.at(index) as FormGroup;\r\n    ruleGroup.get('condition')?.setValue(condition);\r\n  }\r\n\r\n  toggleCondition(): void {\r\n    const currentCondition = this.form.get('condition')?.value;\r\n    const newCondition = currentCondition === 'and' ? 'or' : 'and';\r\n    this.form.get('condition')?.setValue(newCondition);\r\n  }\r\n\r\n  getConditionLabel(): string {\r\n    return this.form.get('condition')?.value === 'and' ? 'AND' : 'OR';\r\n  }\r\n\r\n  resetQuery(): void {\r\n    this.rulesArray.clear();\r\n    this.addRule();\r\n  }\r\n\r\n  applyQuery(): void {\r\n    this.queryChange.emit(this.model);\r\n  }\r\n}\r\n","<div class=\"sa-query-builder\">\r\n  <form [formGroup]=\"form\">\r\n    <div class=\"query-builder-header\">\r\n      <div class=\"where-label\">Where</div>\r\n    </div>\r\n\r\n    <div class=\"query-builder-body\">\r\n      <div class=\"rules-container\" formArrayName=\"rules\">\r\n        @for (rule of rulesArray.controls; track $index) {\r\n          <div class=\"rule-row\" [formGroupName]=\"$index\">\r\n            <!-- Condition selector (And/Or) -->\r\n            @if ($index > 0) {\r\n              <div class=\"condition-selector\">\r\n                <mat-form-field appearance=\"fill\">\r\n                  <mat-select [value]=\"getRowCondition($index)\" (selectionChange)=\"setRowCondition($index, $event.value)\">\r\n                    <mat-option value=\"and\">And</mat-option>\r\n                    <mat-option value=\"or\">Or</mat-option>\r\n                  </mat-select>\r\n                </mat-form-field>\r\n              </div>\r\n            } @else {\r\n              <div class=\"condition-placeholder\"></div>\r\n            }\r\n            \r\n            <!-- Field selection -->\r\n            <mat-form-field class=\"field-select\">\r\n              <mat-label>Select a filter</mat-label>\r\n              <mat-select formControlName=\"field\" (selectionChange)=\"onFieldChange($index, $event.value)\">\r\n                @for (field of config.fields; track field.key) {\r\n                  <mat-option [value]=\"field.key\">\r\n                    {{ field.label }}\r\n                  </mat-option>\r\n                }\r\n              </mat-select>\r\n            </mat-form-field>\r\n\r\n            <!-- Operator selection -->\r\n            <mat-form-field class=\"operator-select\">\r\n              <mat-select formControlName=\"operator\" (selectionChange)=\"onOperatorChange($index, $event.value, rule.value.field)\">\r\n                @for (op of getOperatorsForField(rule.value.field); track op.value) {\r\n                  <mat-option [value]=\"op.value\">\r\n                    {{ op.label }}\r\n                  </mat-option>\r\n                }\r\n              </mat-select>\r\n            </mat-form-field>\r\n\r\n            <!-- Value input based on field type -->\r\n            @switch (getFieldType(rule.value.field)) {\r\n              @case ('string') {\r\n                <mat-form-field class=\"value-input\">\r\n                  <input matInput formControlName=\"value\" placeholder=\"Enter a value...\">\r\n                </mat-form-field>\r\n              }\r\n              @case ('number') {\r\n                <mat-form-field class=\"value-input\">\r\n                  <input matInput type=\"number\" formControlName=\"value\" placeholder=\"Enter a value...\">\r\n                </mat-form-field>\r\n              }\r\n              @case ('date') {\r\n                @if (!isOperatorBetween($index)) {\r\n                  <mat-form-field class=\"value-input\">\r\n                    <input matInput type=\"date\" [matDatepicker]=\"picker\" formControlName=\"value\" (click)=\"picker.open()\">\r\n                    <mat-datepicker-toggle matIconSuffix [for]=\"picker\"></mat-datepicker-toggle>\r\n                    <mat-datepicker #picker></mat-datepicker>\r\n                  </mat-form-field>\r\n                } @else {\r\n                  <div class=\"date-range-container\">\r\n                    <mat-form-field class=\"value-input start-date\">\r\n                      <mat-label>Start date</mat-label>\r\n                      <input matInput [matDatepicker]=\"startPicker\" formControlName=\"valueStart\" (click)=\"startPicker.open()\" (dateChange)=\"onStartDateChange($index, $event)\">\r\n                      <mat-datepicker-toggle matIconSuffix [for]=\"startPicker\"></mat-datepicker-toggle>\r\n                      <mat-datepicker #startPicker></mat-datepicker>\r\n                    </mat-form-field>\r\n                    <mat-form-field class=\"value-input end-date\">\r\n                      <mat-label>End date</mat-label>\r\n                      <input matInput [matDatepicker]=\"endPicker\" formControlName=\"valueEnd\" (click)=\"endPicker.open()\" [min]=\"getMinEndDate($index)\">\r\n                      <mat-datepicker-toggle matIconSuffix [for]=\"endPicker\"></mat-datepicker-toggle>\r\n                      <mat-datepicker #endPicker [startAt]=\"getMinEndDate($index)\"></mat-datepicker>\r\n                    </mat-form-field>\r\n                  </div>\r\n                }\r\n              }\r\n              @case ('boolean') {\r\n                <mat-form-field class=\"value-input\">\r\n                  <mat-select formControlName=\"value\">\r\n                    <mat-option [value]=\"true\">True</mat-option>\r\n                    <mat-option [value]=\"false\">False</mat-option>\r\n                  </mat-select>\r\n                </mat-form-field>\r\n              }\r\n              @case ('select') {\r\n                <mat-form-field class=\"value-input\">\r\n                  <mat-select formControlName=\"value\">\r\n                    @for (option of getFieldOptions(rule.value.field); track option.value) {\r\n                      <mat-option [value]=\"option.value\">\r\n                        {{ option.label }}\r\n                      </mat-option>\r\n                    }\r\n                  </mat-select>\r\n                </mat-form-field>\r\n              }\r\n              @default {\r\n                <mat-form-field class=\"value-input\">\r\n                  <input matInput formControlName=\"value\" placeholder=\"Enter a value...\">\r\n                </mat-form-field>\r\n              }\r\n            }\r\n\r\n            <!-- Remove rule button -->\r\n            <button mat-icon-button class=\"remove-rule-btn\" \r\n                    (click)=\"removeRule($index)\" \r\n                    [disabled]=\"rulesArray.length === 1\"\r\n                    matTooltip=\"Remove filter\">\r\n              <mat-icon>close</mat-icon>\r\n            </button>\r\n          </div>\r\n        }\r\n      </div>\r\n\r\n      <!-- Add rule button -->\r\n      <div class=\"add-rule-container\">\r\n        <button mat-button color=\"primary\" class=\"add-rule-btn\" (click)=\"addRule()\">\r\n          <mat-icon>add</mat-icon> Add a filter\r\n        </button>\r\n      </div>\r\n    </div>\r\n\r\n    <div class=\"query-builder-footer\">\r\n      <button mat-button class=\"reset-btn\" (click)=\"resetQuery()\">Reset</button>\r\n      <button mat-raised-button color=\"primary\" class=\"apply-btn\" (click)=\"applyQuery()\">Apply filters</button>\r\n    </div>\r\n  </form>\r\n</div>\r\n"]}