fx-form-builder-wrapper 2.0.90 → 2.0.92
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/uploader/uploader.component.mjs +36 -13
- package/fesm2022/fx-form-builder-wrapper.mjs +34 -11
- package/fesm2022/fx-form-builder-wrapper.mjs.map +1 -1
- package/package.json +1 -1
- package/esm2022/lib/components/customize-dropdown/customize-dropdown.component.mjs +0 -317
- package/esm2022/lib/components/multiselect-with-form-fields/customize-dropdown.component.mjs +0 -243
- package/esm2022/lib/components/multiselect-with-form-fields/multiselect-dropdown.component.mjs +0 -166
- package/lib/components/customize-dropdown/customize-dropdown.component.d.ts +0 -60
- package/lib/components/multiselect-with-form-fields/customize-dropdown.component.d.ts +0 -59
- package/lib/components/multiselect-with-form-fields/multiselect-dropdown.component.d.ts +0 -51
package/package.json
CHANGED
|
@@ -1,317 +0,0 @@
|
|
|
1
|
-
import { CommonModule } from '@angular/common';
|
|
2
|
-
import { Component } from '@angular/core';
|
|
3
|
-
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
4
|
-
import { FxBaseComponent, FxComponent, FxSelectSetting, FxStringSetting, FxValidatorService } from '@instantsys-labs/fx';
|
|
5
|
-
import { Subject, takeUntil } from 'rxjs';
|
|
6
|
-
import * as i0 from "@angular/core";
|
|
7
|
-
import * as i1 from "@angular/common/http";
|
|
8
|
-
import * as i2 from "../../fx-builder-wrapper.service";
|
|
9
|
-
import * as i3 from "@instantsys-labs/core";
|
|
10
|
-
import * as i4 from "@angular/forms";
|
|
11
|
-
import * as i5 from "@angular/common";
|
|
12
|
-
export class CustomizeDropdownComponent extends FxBaseComponent {
|
|
13
|
-
cdr;
|
|
14
|
-
http;
|
|
15
|
-
fxBuilderWrapperService;
|
|
16
|
-
fxApiService;
|
|
17
|
-
fb;
|
|
18
|
-
eRef;
|
|
19
|
-
destroy$ = new Subject();
|
|
20
|
-
form;
|
|
21
|
-
formObject = {};
|
|
22
|
-
dropdownOpen = false;
|
|
23
|
-
formSubmitted = false;
|
|
24
|
-
findingsOptions = [
|
|
25
|
-
{
|
|
26
|
-
label: 'Proclination',
|
|
27
|
-
value: 'proclination',
|
|
28
|
-
info: 'Forward inclination of teeth',
|
|
29
|
-
selected: false,
|
|
30
|
-
subOptions: [
|
|
31
|
-
{ label: 'Mild', value: 'mild' },
|
|
32
|
-
{ label: 'Moderate', value: 'moderate' },
|
|
33
|
-
{ label: 'Severe', value: 'severe' }
|
|
34
|
-
]
|
|
35
|
-
},
|
|
36
|
-
{
|
|
37
|
-
label: 'Crowding',
|
|
38
|
-
value: 'crowding',
|
|
39
|
-
selected: false,
|
|
40
|
-
subOptions: [
|
|
41
|
-
{ label: 'Mild', value: 'mild' },
|
|
42
|
-
{ label: 'Moderate', value: 'moderate' },
|
|
43
|
-
{ label: 'Severe', value: 'severe' }
|
|
44
|
-
]
|
|
45
|
-
},
|
|
46
|
-
{
|
|
47
|
-
label: 'Spacing',
|
|
48
|
-
value: 'spacing',
|
|
49
|
-
selected: false,
|
|
50
|
-
subOptions: [
|
|
51
|
-
{ label: 'Mild', value: 'mild' },
|
|
52
|
-
{ label: 'Moderate', value: 'moderate' },
|
|
53
|
-
{ label: 'Severe', value: 'severe' }
|
|
54
|
-
]
|
|
55
|
-
},
|
|
56
|
-
{
|
|
57
|
-
label: 'Retroclination',
|
|
58
|
-
value: 'retroclination',
|
|
59
|
-
info: 'Backward inclination of teeth',
|
|
60
|
-
selected: false
|
|
61
|
-
},
|
|
62
|
-
{
|
|
63
|
-
label: 'Rotation',
|
|
64
|
-
value: 'rotation',
|
|
65
|
-
selected: false
|
|
66
|
-
}
|
|
67
|
-
];
|
|
68
|
-
// @HostListener('document:click', ['$event'])
|
|
69
|
-
// onClickOutside(event: MouseEvent) {
|
|
70
|
-
// if (this.dropdownOpen && !this.eRef.nativeElement.contains(event.target)) {
|
|
71
|
-
// this.dropdownOpen = false;
|
|
72
|
-
// this.cdr.detectChanges();
|
|
73
|
-
// }
|
|
74
|
-
// }
|
|
75
|
-
config = {
|
|
76
|
-
displayMode: 'ellipsis',
|
|
77
|
-
placeholderLabel: 'Select Finding'
|
|
78
|
-
};
|
|
79
|
-
constructor(cdr, http, fxBuilderWrapperService, fxApiService, fb, eRef) {
|
|
80
|
-
super(cdr);
|
|
81
|
-
this.cdr = cdr;
|
|
82
|
-
this.http = http;
|
|
83
|
-
this.fxBuilderWrapperService = fxBuilderWrapperService;
|
|
84
|
-
this.fxApiService = fxApiService;
|
|
85
|
-
this.fb = fb;
|
|
86
|
-
this.eRef = eRef;
|
|
87
|
-
this.form = this.fb.group({
|
|
88
|
-
findings: [[]]
|
|
89
|
-
});
|
|
90
|
-
this.onInit.subscribe(() => this._register(this.form));
|
|
91
|
-
}
|
|
92
|
-
ngAfterViewInit() {
|
|
93
|
-
setTimeout(() => {
|
|
94
|
-
const data = [
|
|
95
|
-
{
|
|
96
|
-
label: "Proclination",
|
|
97
|
-
value: "proclination",
|
|
98
|
-
subSelection: {
|
|
99
|
-
label: "Mild",
|
|
100
|
-
value: "mild"
|
|
101
|
-
}
|
|
102
|
-
},
|
|
103
|
-
{
|
|
104
|
-
label: "Overbite",
|
|
105
|
-
value: "overbite",
|
|
106
|
-
subSelection: {
|
|
107
|
-
label: "Moderate",
|
|
108
|
-
value: "moderate"
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
];
|
|
112
|
-
this.patchExistingValues(data);
|
|
113
|
-
}, 2000);
|
|
114
|
-
}
|
|
115
|
-
ngOnInit() {
|
|
116
|
-
this.fxBuilderWrapperService.variables$
|
|
117
|
-
.pipe(takeUntil(this.destroy$))
|
|
118
|
-
.subscribe((variables) => {
|
|
119
|
-
console.log("Variables");
|
|
120
|
-
});
|
|
121
|
-
const serviceUrl = this.fxApiService.getServiceUrl(this.setting('serviceName'));
|
|
122
|
-
this.getOptions(serviceUrl, this.setting('clinicalNotesURL'));
|
|
123
|
-
}
|
|
124
|
-
getOptions(serviceUrl, url) {
|
|
125
|
-
const finalUrl = serviceUrl + url;
|
|
126
|
-
this.http.get(finalUrl).subscribe({
|
|
127
|
-
next: (response) => {
|
|
128
|
-
// Future API logic here
|
|
129
|
-
},
|
|
130
|
-
error: (err) => console.error('Error fetching options', err)
|
|
131
|
-
});
|
|
132
|
-
}
|
|
133
|
-
settings() {
|
|
134
|
-
return [
|
|
135
|
-
new FxSelectSetting({ key: 'displayMode', $title: 'Display Mode', value: 'ellipsis' }, [{ option: 'Ellipsis', value: 'ellipsis' }, { option: 'Compact', value: 'compact' }]),
|
|
136
|
-
new FxStringSetting({ key: 'placeholderLabel', $title: 'Placeholder', value: 'Select Options' }),
|
|
137
|
-
];
|
|
138
|
-
}
|
|
139
|
-
validations() {
|
|
140
|
-
return [FxValidatorService.required];
|
|
141
|
-
}
|
|
142
|
-
/** Dropdown Behavior **/
|
|
143
|
-
toggleDropdown() {
|
|
144
|
-
this.dropdownOpen = !this.dropdownOpen;
|
|
145
|
-
}
|
|
146
|
-
toggleOption(option, event) {
|
|
147
|
-
event.stopPropagation();
|
|
148
|
-
// Toggle checkbox value
|
|
149
|
-
option.selected = !option.selected;
|
|
150
|
-
// Reset radios when unchecked
|
|
151
|
-
if (!option.selected) {
|
|
152
|
-
option.subSelection = null;
|
|
153
|
-
option.touched = false;
|
|
154
|
-
}
|
|
155
|
-
else {
|
|
156
|
-
option.touched = true;
|
|
157
|
-
}
|
|
158
|
-
// ✅ Force UI refresh so radios appear instantly
|
|
159
|
-
this.cdr.detectChanges();
|
|
160
|
-
// Update reactive form
|
|
161
|
-
this.updateFindings();
|
|
162
|
-
}
|
|
163
|
-
/** Form & Label Helpers **/
|
|
164
|
-
get hasSelectedFindings() {
|
|
165
|
-
return this.findingsOptions.some(f => f.selected);
|
|
166
|
-
}
|
|
167
|
-
get selectedFindingsLabel() {
|
|
168
|
-
const selected = this.findingsOptions
|
|
169
|
-
.filter(f => {
|
|
170
|
-
if (f.selected) {
|
|
171
|
-
// If finding has sub-options → only show if a sub-option is selected
|
|
172
|
-
if (f.subOptions?.length) {
|
|
173
|
-
return !!f.subSelection;
|
|
174
|
-
}
|
|
175
|
-
// If no sub-options → always show
|
|
176
|
-
return true;
|
|
177
|
-
}
|
|
178
|
-
return false;
|
|
179
|
-
})
|
|
180
|
-
.map(f => f.label);
|
|
181
|
-
if (selected.length === 0)
|
|
182
|
-
return this.setting('placeholderLabel');
|
|
183
|
-
// Display mode logic (Compact or Ellipsis)
|
|
184
|
-
const maxCount = this.setting('displayMode') === 'compact' ? 2 : 3;
|
|
185
|
-
if (this.setting('displayMode') === 'compact') {
|
|
186
|
-
return selected.length <= maxCount
|
|
187
|
-
? selected.join(', ')
|
|
188
|
-
: `${selected.slice(0, maxCount).join(', ')} +${selected.length - maxCount} more`;
|
|
189
|
-
}
|
|
190
|
-
if (this.setting('displayMode') === 'ellipsis') {
|
|
191
|
-
return selected.length > maxCount
|
|
192
|
-
? `${selected.slice(0, maxCount).join(', ')}, ...`
|
|
193
|
-
: selected.join(', ');
|
|
194
|
-
}
|
|
195
|
-
return selected.join(', ');
|
|
196
|
-
}
|
|
197
|
-
/** Update Findings + Validation **/
|
|
198
|
-
// updateFindings() {
|
|
199
|
-
// const selected = this.findingsOptions
|
|
200
|
-
// .filter(f => {
|
|
201
|
-
// if (f.selected) {
|
|
202
|
-
// // Only include in final form if:
|
|
203
|
-
// // - no subOptions, or
|
|
204
|
-
// // - subOptions with valid subSelection
|
|
205
|
-
// if (f.subOptions?.length) {
|
|
206
|
-
// return !!f.subSelection;
|
|
207
|
-
// }
|
|
208
|
-
// return true;
|
|
209
|
-
// }
|
|
210
|
-
// return false;
|
|
211
|
-
// })
|
|
212
|
-
// .map(f => {
|
|
213
|
-
// const sub = f.subOptions?.find(s => s.value === f.subSelection) || null;
|
|
214
|
-
// return {
|
|
215
|
-
// label: f.label,
|
|
216
|
-
// value: f.value,
|
|
217
|
-
// subSelection: sub ? { label: sub.label, value: sub.value } : null
|
|
218
|
-
// };
|
|
219
|
-
// });
|
|
220
|
-
// // Update reactive form value
|
|
221
|
-
// this.form.patchValue({ findings: selected }, { emitEvent: false });
|
|
222
|
-
// // Validation logic remains same
|
|
223
|
-
// const invalidItems = this.findingsOptions.filter(
|
|
224
|
-
// f => f.selected && f.subOptions && !f.subSelection
|
|
225
|
-
// );
|
|
226
|
-
// this.form.get('findings')?.setErrors(
|
|
227
|
-
// invalidItems.length > 0 ? { missingSubSelection: true } : null
|
|
228
|
-
// );
|
|
229
|
-
// }
|
|
230
|
-
updateFindings() {
|
|
231
|
-
// Filter selected options with valid subSelection (if subOptions exist)
|
|
232
|
-
const selected = this.findingsOptions
|
|
233
|
-
.filter(f => {
|
|
234
|
-
if (f.selected) {
|
|
235
|
-
// Only include in final form if:
|
|
236
|
-
// - no subOptions, or
|
|
237
|
-
// - subOptions with valid subSelection
|
|
238
|
-
if (f.subOptions?.length) {
|
|
239
|
-
return !!f.subSelection;
|
|
240
|
-
}
|
|
241
|
-
return true;
|
|
242
|
-
}
|
|
243
|
-
return false;
|
|
244
|
-
})
|
|
245
|
-
.map(f => {
|
|
246
|
-
const sub = f.subOptions?.find(s => s.value === f.subSelection) || null;
|
|
247
|
-
return {
|
|
248
|
-
label: f.label,
|
|
249
|
-
value: f.value,
|
|
250
|
-
subSelection: sub ? { label: sub.label, value: sub.value } : null
|
|
251
|
-
};
|
|
252
|
-
});
|
|
253
|
-
// Update reactive form value
|
|
254
|
-
this.form.patchValue({ findings: selected }, { emitEvent: false });
|
|
255
|
-
// Validation logic:
|
|
256
|
-
// Check if there are any selected options with subOptions but without subSelection
|
|
257
|
-
const invalidItems = this.findingsOptions.filter(f => f.selected && f.subOptions && !f.subSelection);
|
|
258
|
-
// If there are any invalid items, mark the form as invalid with a custom error
|
|
259
|
-
if (invalidItems.length > 0) {
|
|
260
|
-
this.form.get('findings')?.setErrors({ missingSubSelection: true });
|
|
261
|
-
}
|
|
262
|
-
else {
|
|
263
|
-
// Clear the error if everything is valid
|
|
264
|
-
this.form.get('findings')?.setErrors(null);
|
|
265
|
-
}
|
|
266
|
-
}
|
|
267
|
-
onSubmit() {
|
|
268
|
-
this.formSubmitted = true;
|
|
269
|
-
if (this.form.invalid) {
|
|
270
|
-
console.warn('⚠️ Please select a sub-option for all selected findings with sub-options.');
|
|
271
|
-
return;
|
|
272
|
-
}
|
|
273
|
-
console.log('✅ Form Value:', this.form.value);
|
|
274
|
-
}
|
|
275
|
-
// patchExistingValues(data: any[]) {
|
|
276
|
-
// this.findingsOptions.forEach(opt => {
|
|
277
|
-
// const match = data.find(x => x.value === opt.value);
|
|
278
|
-
// opt.selected = !!match;
|
|
279
|
-
// opt.subSelection = match?.severity?.value || null;
|
|
280
|
-
// });
|
|
281
|
-
// this.updateFindings();
|
|
282
|
-
// }
|
|
283
|
-
patchExistingValues(data) {
|
|
284
|
-
// Iterate through the findingsOptions and find the corresponding option for each entry in data
|
|
285
|
-
this.findingsOptions.forEach(opt => {
|
|
286
|
-
const match = data.find(x => x.value === opt.value);
|
|
287
|
-
if (match) {
|
|
288
|
-
opt.selected = true;
|
|
289
|
-
opt.subSelection = match.subSelection ? match.subSelection.value : null;
|
|
290
|
-
console.log("Matched Option:", opt);
|
|
291
|
-
console.log("SubSelection Set To:", this.findingsOptions);
|
|
292
|
-
}
|
|
293
|
-
else {
|
|
294
|
-
opt.selected = false;
|
|
295
|
-
opt.subSelection = null;
|
|
296
|
-
}
|
|
297
|
-
});
|
|
298
|
-
// Manually trigger change detection if needed
|
|
299
|
-
this.cdr.detectChanges();
|
|
300
|
-
this.updateFindings();
|
|
301
|
-
}
|
|
302
|
-
selectSubOption(option, subOption) {
|
|
303
|
-
// Set the subSelection to the clicked value
|
|
304
|
-
option.subSelection = subOption.value;
|
|
305
|
-
// Mark the option as touched, which will help in form validation
|
|
306
|
-
option.touched = true;
|
|
307
|
-
// Update the form state
|
|
308
|
-
this.updateFindings();
|
|
309
|
-
}
|
|
310
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomizeDropdownComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.HttpClient }, { token: i2.FxBuilderWrapperService }, { token: i3.ApiServiceRegistry }, { token: i4.FormBuilder }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
311
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CustomizeDropdownComponent, isStandalone: true, selector: "lib-customize-dropdown", usesInheritance: true, ngImport: i0, template: "<fx-component [fxData]=\"fxData\" #fxComponent>\r\n <div class=\"container\">\r\n <form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\" class=\"relative\">\r\n <!-- Header -->\r\n <div #dropdownWrapper class=\"relative w-80\">\r\n <button\r\n type=\"button\"\r\n class=\"w-full border border-gray-300 rounded-md px-3 py-2 flex justify-between items-center bg-white text-gray-700 hover:border-blue-400\"\r\n (click)=\"toggleDropdown()\"\r\n >\r\n <span *ngIf=\"hasSelectedFindings; else placeholder\">\r\n {{ selectedFindingsLabel }}\r\n </span>\r\n <ng-template #placeholder>\r\n {{ setting('placeholderLabel') }}\r\n </ng-template>\r\n <svg\r\n class=\"w-5 h-5 ml-2 text-gray-500 transform transition-transform duration-200\"\r\n [class.rotate-180]=\"dropdownOpen\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n viewBox=\"0 0 24 24\"\r\n >\r\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M19 9l-7 7-7-7\" />\r\n </svg>\r\n </button>\r\n\r\n <!-- Panel -->\r\n <div\r\n *ngIf=\"dropdownOpen\"\r\n class=\"absolute mt-1 w-full bg-white border border-gray-300 rounded-md shadow-lg max-h-64 overflow-y-auto z-10\"\r\n >\r\n <div\r\n *ngFor=\"let option of findingsOptions\"\r\n class=\"border-b border-gray-100 last:border-none p-2 hover:bg-gray-50 cursor-pointer\"\r\n (click)=\"$event.stopPropagation()\"\r\n >\r\n <!-- Checkbox + Label -->\r\n <div class=\"flex items-center gap-2\">\r\n <input\r\n type=\"checkbox\"\r\n class=\"w-4 h-4 text-blue-500 border-gray-300 rounded cursor-pointer\"\r\n [(ngModel)]=\"option.selected\"\r\n [ngModelOptions]=\"{ standalone: true }\"\r\n (click)=\"toggleOption(option, $event)\"\r\n />\r\n <label\r\n class=\"text-gray-800 font-medium cursor-pointer select-none\"\r\n (click)=\"toggleOption(option, $event)\"\r\n >\r\n {{ option.label }}\r\n </label>\r\n\r\n <span\r\n *ngIf=\"option.info\"\r\n class=\"ml-auto text-blue-500 text-sm cursor-pointer\"\r\n title=\"{{ option.info }}\"\r\n >\r\n \u24D8\r\n </span>\r\n </div>\r\n\r\n <!-- Radios -->\r\n <div\r\n class=\"flex items-center flex-wrap gap-4 ml-6 mt-2 text-sm\"\r\n *ngIf=\"option.selected && option.subOptions\"\r\n >\r\n <ng-container *ngFor=\"let s of option.subOptions\">\r\n <label\r\n class=\"flex items-center space-x-1 cursor-pointer\"\r\n (click)=\"selectSubOption(option, s)\"\r\n >\r\n <input\r\n type=\"radio\"\r\n [name]=\"option.value\"\r\n [value]=\"s.value\"\r\n [(ngModel)]=\"option.subSelection\"\r\n [ngModelOptions]=\"{ standalone: true }\"\r\n class=\"text-blue-600 cursor-pointer\"\r\n />\r\n <span>{{ s.label }}</span>\r\n </label>\r\n</ng-container>\r\n\r\n <!-- Validation Message -->\r\n <div\r\n *ngIf=\"option.selected && option.subOptions && !option.subSelection\"\r\n class=\"text-red-500 text-xs mt-1 w-full ml-1\"\r\n >\r\n Please select one option\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n </form>\r\n </div>\r\n</fx-component>\r\n", styles: [".container{width:300px}.dropdown{position:relative;-webkit-user-select:none;user-select:none}.dropdown-header{border:1px solid #ccc;border-radius:4px;padding:6px 8px;background-color:#fff;cursor:pointer;display:flex;justify-content:space-between;align-items:center}.dropdown-panel{position:absolute;width:100%;background:#fff;border:1px solid #ddd;margin-top:4px;border-radius:4px;max-height:250px;overflow-y:auto;z-index:1000;box-shadow:0 2px 8px #0000001a}.dropdown-item{padding:4px 8px;border-bottom:1px solid #f1f1f1}.dropdown-item:last-child{border-bottom:none}.item-header{display:flex;align-items:center;gap:6px}.label{cursor:pointer;flex-grow:1}.info{margin-left:auto;cursor:help;font-size:12px;color:#666}.sub-options{display:flex;align-items:center;flex-wrap:wrap;padding-left:22px;font-size:13px;margin-top:4px}.error{color:#e63946;font-size:11px;margin-left:22px;margin-top:2px}.arrow{font-size:10px;color:#555}.submit-btn{margin-top:1rem;padding:6px 12px;background:#007bff;color:#fff;border:none;border-radius:4px;cursor:pointer}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i4.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i4.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: i4.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i4.RadioControlValueAccessor, selector: "input[type=radio][formControlName],input[type=radio][formControl],input[type=radio][ngModel]", inputs: ["name", "formControlName", "value"] }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: FxComponent, selector: "fx-component", inputs: ["fxData"] }] });
|
|
312
|
-
}
|
|
313
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomizeDropdownComponent, decorators: [{
|
|
314
|
-
type: Component,
|
|
315
|
-
args: [{ selector: 'lib-customize-dropdown', standalone: true, imports: [CommonModule, ReactiveFormsModule, FormsModule, FxComponent], template: "<fx-component [fxData]=\"fxData\" #fxComponent>\r\n <div class=\"container\">\r\n <form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\" class=\"relative\">\r\n <!-- Header -->\r\n <div #dropdownWrapper class=\"relative w-80\">\r\n <button\r\n type=\"button\"\r\n class=\"w-full border border-gray-300 rounded-md px-3 py-2 flex justify-between items-center bg-white text-gray-700 hover:border-blue-400\"\r\n (click)=\"toggleDropdown()\"\r\n >\r\n <span *ngIf=\"hasSelectedFindings; else placeholder\">\r\n {{ selectedFindingsLabel }}\r\n </span>\r\n <ng-template #placeholder>\r\n {{ setting('placeholderLabel') }}\r\n </ng-template>\r\n <svg\r\n class=\"w-5 h-5 ml-2 text-gray-500 transform transition-transform duration-200\"\r\n [class.rotate-180]=\"dropdownOpen\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n viewBox=\"0 0 24 24\"\r\n >\r\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M19 9l-7 7-7-7\" />\r\n </svg>\r\n </button>\r\n\r\n <!-- Panel -->\r\n <div\r\n *ngIf=\"dropdownOpen\"\r\n class=\"absolute mt-1 w-full bg-white border border-gray-300 rounded-md shadow-lg max-h-64 overflow-y-auto z-10\"\r\n >\r\n <div\r\n *ngFor=\"let option of findingsOptions\"\r\n class=\"border-b border-gray-100 last:border-none p-2 hover:bg-gray-50 cursor-pointer\"\r\n (click)=\"$event.stopPropagation()\"\r\n >\r\n <!-- Checkbox + Label -->\r\n <div class=\"flex items-center gap-2\">\r\n <input\r\n type=\"checkbox\"\r\n class=\"w-4 h-4 text-blue-500 border-gray-300 rounded cursor-pointer\"\r\n [(ngModel)]=\"option.selected\"\r\n [ngModelOptions]=\"{ standalone: true }\"\r\n (click)=\"toggleOption(option, $event)\"\r\n />\r\n <label\r\n class=\"text-gray-800 font-medium cursor-pointer select-none\"\r\n (click)=\"toggleOption(option, $event)\"\r\n >\r\n {{ option.label }}\r\n </label>\r\n\r\n <span\r\n *ngIf=\"option.info\"\r\n class=\"ml-auto text-blue-500 text-sm cursor-pointer\"\r\n title=\"{{ option.info }}\"\r\n >\r\n \u24D8\r\n </span>\r\n </div>\r\n\r\n <!-- Radios -->\r\n <div\r\n class=\"flex items-center flex-wrap gap-4 ml-6 mt-2 text-sm\"\r\n *ngIf=\"option.selected && option.subOptions\"\r\n >\r\n <ng-container *ngFor=\"let s of option.subOptions\">\r\n <label\r\n class=\"flex items-center space-x-1 cursor-pointer\"\r\n (click)=\"selectSubOption(option, s)\"\r\n >\r\n <input\r\n type=\"radio\"\r\n [name]=\"option.value\"\r\n [value]=\"s.value\"\r\n [(ngModel)]=\"option.subSelection\"\r\n [ngModelOptions]=\"{ standalone: true }\"\r\n class=\"text-blue-600 cursor-pointer\"\r\n />\r\n <span>{{ s.label }}</span>\r\n </label>\r\n</ng-container>\r\n\r\n <!-- Validation Message -->\r\n <div\r\n *ngIf=\"option.selected && option.subOptions && !option.subSelection\"\r\n class=\"text-red-500 text-xs mt-1 w-full ml-1\"\r\n >\r\n Please select one option\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n </form>\r\n </div>\r\n</fx-component>\r\n", styles: [".container{width:300px}.dropdown{position:relative;-webkit-user-select:none;user-select:none}.dropdown-header{border:1px solid #ccc;border-radius:4px;padding:6px 8px;background-color:#fff;cursor:pointer;display:flex;justify-content:space-between;align-items:center}.dropdown-panel{position:absolute;width:100%;background:#fff;border:1px solid #ddd;margin-top:4px;border-radius:4px;max-height:250px;overflow-y:auto;z-index:1000;box-shadow:0 2px 8px #0000001a}.dropdown-item{padding:4px 8px;border-bottom:1px solid #f1f1f1}.dropdown-item:last-child{border-bottom:none}.item-header{display:flex;align-items:center;gap:6px}.label{cursor:pointer;flex-grow:1}.info{margin-left:auto;cursor:help;font-size:12px;color:#666}.sub-options{display:flex;align-items:center;flex-wrap:wrap;padding-left:22px;font-size:13px;margin-top:4px}.error{color:#e63946;font-size:11px;margin-left:22px;margin-top:2px}.arrow{font-size:10px;color:#555}.submit-btn{margin-top:1rem;padding:6px 12px;background:#007bff;color:#fff;border:none;border-radius:4px;cursor:pointer}\n"] }]
|
|
316
|
-
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1.HttpClient }, { type: i2.FxBuilderWrapperService }, { type: i3.ApiServiceRegistry }, { type: i4.FormBuilder }, { type: i0.ElementRef }] });
|
|
317
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"customize-dropdown.component.js","sourceRoot":"","sources":["../../../../../../projects/fx-builder-wrapper/src/lib/components/customize-dropdown/customize-dropdown.component.ts","../../../../../../projects/fx-builder-wrapper/src/lib/components/customize-dropdown/customize-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { AfterViewInit, ChangeDetectorRef, Component, ElementRef, HostListener, OnInit } from '@angular/core';\r\nimport { FormBuilder, FormGroup, FormsModule, ReactiveFormsModule } from '@angular/forms';\r\nimport { ApiServiceRegistry } from '@instantsys-labs/core';\r\nimport {\r\n  FxBaseComponent,\r\n  FxComponent,\r\n  FxSelectSetting,\r\n  FxSetting,\r\n  FxStringSetting,\r\n  FxValidation,\r\n  FxValidatorService\r\n} from '@instantsys-labs/fx';\r\nimport { FxBuilderWrapperService } from '../../fx-builder-wrapper.service';\r\nimport { Subject, takeUntil } from 'rxjs';\r\n\r\ninterface SubOption {\r\n  label: string;\r\n  value: string;\r\n}\r\n\r\ninterface CustomDropdownOption {\r\n  label: string;\r\n  value: string;\r\n  info?: string;\r\n  selected: boolean;\r\n  subOptions?: SubOption[];\r\n  subSelection?: string | null;\r\n  disabled?: boolean;\r\n  touched?: boolean;\r\n}\r\n\r\ninterface CustomDropdownConfig {\r\n  displayMode?: 'compact' | 'ellipsis';\r\n  placeholderLabel?: string;\r\n}\r\n\r\n@Component({\r\n  selector: 'lib-customize-dropdown',\r\n  standalone: true,\r\n  imports: [CommonModule, ReactiveFormsModule, FormsModule, FxComponent],\r\n  templateUrl: './customize-dropdown.component.html',\r\n  styleUrls: ['./customize-dropdown.component.css']\r\n})\r\nexport class CustomizeDropdownComponent extends FxBaseComponent implements OnInit , AfterViewInit{\r\n  private destroy$ = new Subject<boolean>();\r\n  form!: FormGroup;\r\n  formObject: object = {};\r\n  dropdownOpen = false;\r\n  formSubmitted = false;\r\n\r\n  findingsOptions: CustomDropdownOption[] = [\r\n    {\r\n      label: 'Proclination',\r\n      value: 'proclination',\r\n      info: 'Forward inclination of teeth',\r\n      selected: false,\r\n      subOptions: [\r\n        { label: 'Mild', value: 'mild' },\r\n        { label: 'Moderate', value: 'moderate' },\r\n        { label: 'Severe', value: 'severe' }\r\n      ]\r\n    },\r\n    {\r\n      label: 'Crowding',\r\n      value: 'crowding',\r\n      selected: false,\r\n      subOptions: [\r\n        { label: 'Mild', value: 'mild' },\r\n        { label: 'Moderate', value: 'moderate' },\r\n        { label: 'Severe', value: 'severe' }\r\n      ]\r\n    },\r\n    {\r\n      label: 'Spacing',\r\n      value: 'spacing',\r\n      selected: false,\r\n      subOptions: [\r\n        { label: 'Mild', value: 'mild' },\r\n        { label: 'Moderate', value: 'moderate' },\r\n        { label: 'Severe', value: 'severe' }\r\n      ]\r\n    },\r\n    {\r\n      label: 'Retroclination',\r\n      value: 'retroclination',\r\n      info: 'Backward inclination of teeth',\r\n      selected: false\r\n    },\r\n    {\r\n      label: 'Rotation',\r\n      value: 'rotation',\r\n      selected: false\r\n    }\r\n  ];\r\n\r\n  // @HostListener('document:click', ['$event'])\r\n  // onClickOutside(event: MouseEvent) {\r\n  //   if (this.dropdownOpen && !this.eRef.nativeElement.contains(event.target)) {\r\n  //     this.dropdownOpen = false;\r\n  //     this.cdr.detectChanges();\r\n  //   }\r\n  // }\r\n\r\n  config: CustomDropdownConfig = {\r\n    displayMode: 'ellipsis',\r\n    placeholderLabel: 'Select Finding'\r\n  };\r\n\r\n  constructor(\r\n    private cdr: ChangeDetectorRef,\r\n    private http: HttpClient,\r\n    private fxBuilderWrapperService: FxBuilderWrapperService,\r\n    private fxApiService: ApiServiceRegistry,\r\n    private fb: FormBuilder,\r\n    private eRef: ElementRef\r\n  ) {\r\n    super(cdr);\r\n    this.form = this.fb.group({\r\n      findings: [[]]\r\n    });\r\n    this.onInit.subscribe(() => this._register(this.form));\r\n  }\r\n  ngAfterViewInit(): void {\r\n    setTimeout(()=>{\r\n      const data = [\r\n  {\r\n    label: \"Proclination\",\r\n    value: \"proclination\",\r\n    subSelection: {\r\n      label: \"Mild\",\r\n      value: \"mild\"\r\n    }\r\n  },\r\n  {\r\n    label: \"Overbite\",\r\n    value: \"overbite\",\r\n    subSelection: {\r\n      label: \"Moderate\",\r\n      value: \"moderate\"\r\n    }\r\n  }\r\n];\r\nthis.patchExistingValues(data);\r\n    },2000);\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    this.fxBuilderWrapperService.variables$\r\n      .pipe(takeUntil(this.destroy$))\r\n      .subscribe((variables: any) => {\r\n        console.log(\"Variables\",)\r\n      });\r\n    const serviceUrl = this.fxApiService.getServiceUrl(this.setting('serviceName'));\r\n    this.getOptions(serviceUrl, this.setting('clinicalNotesURL'));\r\n  }\r\n\r\n  getOptions(serviceUrl: string, url: string) {\r\n    const finalUrl = serviceUrl + url;\r\n    this.http.get<any[]>(finalUrl).subscribe({\r\n      next: (response: any) => {\r\n        // Future API logic here\r\n      },\r\n      error: (err) => console.error('Error fetching options', err)\r\n    });\r\n  }\r\n\r\n  protected settings(): FxSetting[] {\r\n    return [\r\n      new FxSelectSetting({ key: 'displayMode', $title: 'Display Mode', value: 'ellipsis' }, [{ option: 'Ellipsis', value: 'ellipsis' }, { option: 'Compact', value: 'compact' }]),\r\n      new FxStringSetting({ key: 'placeholderLabel', $title: 'Placeholder', value: 'Select Options' }),\r\n    ];\r\n  }\r\n\r\n  protected validations(): FxValidation[] {\r\n    return [FxValidatorService.required];\r\n  }\r\n\r\n  /** Dropdown Behavior **/\r\n  toggleDropdown() {\r\n    this.dropdownOpen = !this.dropdownOpen;\r\n  }\r\n\r\n  toggleOption(option: CustomDropdownOption, event: Event) {\r\n    event.stopPropagation();\r\n\r\n    // Toggle checkbox value\r\n    option.selected = !option.selected;\r\n\r\n    // Reset radios when unchecked\r\n    if (!option.selected) {\r\n      option.subSelection = null;\r\n      option.touched = false;\r\n    }\r\n\r\n    else{\r\n      option.touched = true;\r\n    }\r\n\r\n    // ✅ Force UI refresh so radios appear instantly\r\n    this.cdr.detectChanges();\r\n\r\n    // Update reactive form\r\n    this.updateFindings();\r\n  }\r\n\r\n  /** Form & Label Helpers **/\r\n  get hasSelectedFindings(): boolean {\r\n    return this.findingsOptions.some(f => f.selected);\r\n  }\r\n\r\n  get selectedFindingsLabel(): string {\r\n    const selected = this.findingsOptions\r\n      .filter(f => {\r\n        if (f.selected) {\r\n          // If finding has sub-options → only show if a sub-option is selected\r\n          if (f.subOptions?.length) {\r\n            return !!f.subSelection;\r\n          }\r\n          // If no sub-options → always show\r\n          return true;\r\n        }\r\n        return false;\r\n      })\r\n      .map(f => f.label);\r\n\r\n    if (selected.length === 0) return this.setting('placeholderLabel');\r\n\r\n    // Display mode logic (Compact or Ellipsis)\r\n    const maxCount = this.setting('displayMode') === 'compact' ? 2 : 3;\r\n    if (this.setting('displayMode') === 'compact') {\r\n      return selected.length <= maxCount\r\n        ? selected.join(', ')\r\n        : `${selected.slice(0, maxCount).join(', ')} +${selected.length - maxCount} more`;\r\n    }\r\n\r\n    if (this.setting('displayMode') === 'ellipsis') {\r\n      return selected.length > maxCount\r\n        ? `${selected.slice(0, maxCount).join(', ')}, ...`\r\n        : selected.join(', ');\r\n    }\r\n\r\n    return selected.join(', ');\r\n  }\r\n\r\n  /** Update Findings + Validation **/\r\n  // updateFindings() {\r\n  //   const selected = this.findingsOptions\r\n  //     .filter(f => {\r\n  //       if (f.selected) {\r\n  //         // Only include in final form if:\r\n  //         //  - no subOptions, or\r\n  //         //  - subOptions with valid subSelection\r\n  //         if (f.subOptions?.length) {\r\n  //           return !!f.subSelection;\r\n  //         }\r\n  //         return true;\r\n  //       }\r\n  //       return false;\r\n  //     })\r\n  //     .map(f => {\r\n  //       const sub = f.subOptions?.find(s => s.value === f.subSelection) || null;\r\n  //       return {\r\n  //         label: f.label,\r\n  //         value: f.value,\r\n  //         subSelection: sub ? { label: sub.label, value: sub.value } : null\r\n  //       };\r\n  //     });\r\n\r\n  //   // Update reactive form value\r\n  //   this.form.patchValue({ findings: selected }, { emitEvent: false });\r\n\r\n  //   // Validation logic remains same\r\n  //   const invalidItems = this.findingsOptions.filter(\r\n  //     f => f.selected && f.subOptions && !f.subSelection\r\n  //   );\r\n  //   this.form.get('findings')?.setErrors(\r\n  //     invalidItems.length > 0 ? { missingSubSelection: true } : null\r\n  //   );\r\n  // }\r\n\r\n  updateFindings() {\r\n  // Filter selected options with valid subSelection (if subOptions exist)\r\n  const selected = this.findingsOptions\r\n    .filter(f => {\r\n      if (f.selected) {\r\n        // Only include in final form if:\r\n        //  - no subOptions, or\r\n        //  - subOptions with valid subSelection\r\n        if (f.subOptions?.length) {\r\n          return !!f.subSelection;\r\n        }\r\n        return true;\r\n      }\r\n      return false;\r\n    })\r\n    .map(f => {\r\n      const sub = f.subOptions?.find(s => s.value === f.subSelection) || null;\r\n      return {\r\n        label: f.label,\r\n        value: f.value,\r\n        subSelection: sub ? { label: sub.label, value: sub.value } : null\r\n      };\r\n    });\r\n\r\n  // Update reactive form value\r\n  this.form.patchValue({ findings: selected }, { emitEvent: false });\r\n\r\n  // Validation logic:\r\n  // Check if there are any selected options with subOptions but without subSelection\r\n  const invalidItems = this.findingsOptions.filter(\r\n    f => f.selected && f.subOptions && !f.subSelection\r\n  );\r\n\r\n  // If there are any invalid items, mark the form as invalid with a custom error\r\n  if (invalidItems.length > 0) {\r\n    this.form.get('findings')?.setErrors({ missingSubSelection: true });\r\n  } else {\r\n    // Clear the error if everything is valid\r\n    this.form.get('findings')?.setErrors(null);\r\n  }\r\n}\r\n\r\n\r\n  onSubmit() {\r\n    this.formSubmitted = true;\r\n    if (this.form.invalid) {\r\n      console.warn('⚠️ Please select a sub-option for all selected findings with sub-options.');\r\n      return;\r\n    }\r\n    console.log('✅ Form Value:', this.form.value);\r\n  }\r\n\r\n  // patchExistingValues(data: any[]) {\r\n  //   this.findingsOptions.forEach(opt => {\r\n  //     const match = data.find(x => x.value === opt.value);\r\n  //     opt.selected = !!match;\r\n  //     opt.subSelection = match?.severity?.value || null;\r\n  //   });\r\n  //   this.updateFindings();\r\n  // }\r\n\r\npatchExistingValues(data: any[]) {\r\n  // Iterate through the findingsOptions and find the corresponding option for each entry in data\r\n  this.findingsOptions.forEach(opt => {\r\n    const match = data.find(x => x.value === opt.value);\r\n    \r\n    if (match) {\r\n      opt.selected = true;\r\n      opt.subSelection = match.subSelection ? match.subSelection.value : null;\r\n\r\n      console.log(\"Matched Option:\", opt);\r\n      console.log(\"SubSelection Set To:\", this.findingsOptions);\r\n    } else {\r\n      opt.selected = false;\r\n      opt.subSelection = null;\r\n    }\r\n  });\r\n\r\n  // Manually trigger change detection if needed\r\n  this.cdr.detectChanges();\r\n  \r\n  this.updateFindings();\r\n}\r\n\r\nselectSubOption(option: any, subOption: any) {\r\n  // Set the subSelection to the clicked value\r\n  option.subSelection = subOption.value;\r\n  // Mark the option as touched, which will help in form validation\r\n  option.touched = true;\r\n  // Update the form state\r\n  this.updateFindings();\r\n}\r\n\r\n}\r\n","<fx-component [fxData]=\"fxData\" #fxComponent>\r\n  <div class=\"container\">\r\n    <form [formGroup]=\"form\" (ngSubmit)=\"onSubmit()\" class=\"relative\">\r\n      <!-- Header -->\r\n      <div #dropdownWrapper class=\"relative w-80\">\r\n        <button\r\n          type=\"button\"\r\n          class=\"w-full border border-gray-300 rounded-md px-3 py-2 flex justify-between items-center bg-white text-gray-700 hover:border-blue-400\"\r\n          (click)=\"toggleDropdown()\"\r\n        >\r\n          <span *ngIf=\"hasSelectedFindings; else placeholder\">\r\n            {{ selectedFindingsLabel }}\r\n          </span>\r\n          <ng-template #placeholder>\r\n            {{ setting('placeholderLabel') }}\r\n          </ng-template>\r\n          <svg\r\n            class=\"w-5 h-5 ml-2 text-gray-500 transform transition-transform duration-200\"\r\n            [class.rotate-180]=\"dropdownOpen\"\r\n            fill=\"none\"\r\n            stroke=\"currentColor\"\r\n            viewBox=\"0 0 24 24\"\r\n          >\r\n            <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M19 9l-7 7-7-7\" />\r\n          </svg>\r\n        </button>\r\n\r\n        <!-- Panel -->\r\n        <div\r\n          *ngIf=\"dropdownOpen\"\r\n          class=\"absolute mt-1 w-full bg-white border border-gray-300 rounded-md shadow-lg max-h-64 overflow-y-auto z-10\"\r\n        >\r\n          <div\r\n            *ngFor=\"let option of findingsOptions\"\r\n            class=\"border-b border-gray-100 last:border-none p-2 hover:bg-gray-50 cursor-pointer\"\r\n            (click)=\"$event.stopPropagation()\"\r\n          >\r\n            <!-- Checkbox + Label -->\r\n            <div class=\"flex items-center gap-2\">\r\n              <input\r\n                type=\"checkbox\"\r\n                class=\"w-4 h-4 text-blue-500 border-gray-300 rounded cursor-pointer\"\r\n                [(ngModel)]=\"option.selected\"\r\n                [ngModelOptions]=\"{ standalone: true }\"\r\n                (click)=\"toggleOption(option, $event)\"\r\n              />\r\n              <label\r\n                class=\"text-gray-800 font-medium cursor-pointer select-none\"\r\n                (click)=\"toggleOption(option, $event)\"\r\n              >\r\n                {{ option.label }}\r\n              </label>\r\n\r\n              <span\r\n                *ngIf=\"option.info\"\r\n                class=\"ml-auto text-blue-500 text-sm cursor-pointer\"\r\n                title=\"{{ option.info }}\"\r\n              >\r\n                ⓘ\r\n              </span>\r\n            </div>\r\n\r\n            <!-- Radios -->\r\n            <div\r\n              class=\"flex items-center flex-wrap gap-4 ml-6 mt-2 text-sm\"\r\n              *ngIf=\"option.selected && option.subOptions\"\r\n            >\r\n              <ng-container *ngFor=\"let s of option.subOptions\">\r\n  <label\r\n    class=\"flex items-center space-x-1 cursor-pointer\"\r\n    (click)=\"selectSubOption(option, s)\"\r\n  >\r\n    <input\r\n      type=\"radio\"\r\n      [name]=\"option.value\"\r\n      [value]=\"s.value\"\r\n      [(ngModel)]=\"option.subSelection\"\r\n      [ngModelOptions]=\"{ standalone: true }\"\r\n      class=\"text-blue-600 cursor-pointer\"\r\n    />\r\n    <span>{{ s.label }}</span>\r\n  </label>\r\n</ng-container>\r\n\r\n              <!-- Validation Message -->\r\n              <div\r\n                *ngIf=\"option.selected && option.subOptions && !option.subSelection\"\r\n                class=\"text-red-500 text-xs mt-1 w-full ml-1\"\r\n              >\r\n                Please select one option\r\n              </div>\r\n            </div>\r\n          </div>\r\n        </div>\r\n      </div>\r\n    </form>\r\n  </div>\r\n</fx-component>\r\n"]}
|