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/esm2022/lib/components/multiselect-with-form-fields/customize-dropdown.component.mjs
DELETED
|
@@ -1,243 +0,0 @@
|
|
|
1
|
-
import { CommonModule } from '@angular/common';
|
|
2
|
-
import { Component } from '@angular/core';
|
|
3
|
-
import { FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';
|
|
4
|
-
import { Subject, takeUntil } from 'rxjs';
|
|
5
|
-
import { FxBaseComponent, FxComponent, FxSelectSetting, FxStringSetting, FxValidatorService } from '@instantsys-labs/fx';
|
|
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
|
-
config = {
|
|
69
|
-
displayMode: 'ellipsis',
|
|
70
|
-
placeholderLabel: 'Select Finding'
|
|
71
|
-
};
|
|
72
|
-
constructor(cdr, http, fxBuilderWrapperService, fxApiService, fb, eRef) {
|
|
73
|
-
super(cdr);
|
|
74
|
-
this.cdr = cdr;
|
|
75
|
-
this.http = http;
|
|
76
|
-
this.fxBuilderWrapperService = fxBuilderWrapperService;
|
|
77
|
-
this.fxApiService = fxApiService;
|
|
78
|
-
this.fb = fb;
|
|
79
|
-
this.eRef = eRef;
|
|
80
|
-
this.form = this.fb.group({
|
|
81
|
-
findings: this.fb.array([]) // Initially an empty array
|
|
82
|
-
});
|
|
83
|
-
// Initialize the form array
|
|
84
|
-
this.updateFindingsArray();
|
|
85
|
-
this.onInit.subscribe(() => this._register(this.form));
|
|
86
|
-
}
|
|
87
|
-
ngAfterViewInit() {
|
|
88
|
-
setTimeout(() => {
|
|
89
|
-
const data = [
|
|
90
|
-
{
|
|
91
|
-
"value": "proclination",
|
|
92
|
-
"subSelection": { "label": "Mild", "value": "mild" }
|
|
93
|
-
},
|
|
94
|
-
];
|
|
95
|
-
this.patchExistingValues(data);
|
|
96
|
-
}, 2000);
|
|
97
|
-
}
|
|
98
|
-
updateFindingsArray() {
|
|
99
|
-
const findingsFormArray = this.form.get('findings');
|
|
100
|
-
findingsFormArray.clear();
|
|
101
|
-
this.findingsOptions.forEach((option) => {
|
|
102
|
-
if (option.selected) {
|
|
103
|
-
findingsFormArray.push(this.fb.group({
|
|
104
|
-
value: [option.value],
|
|
105
|
-
subSelection: [option.subSelection]
|
|
106
|
-
}));
|
|
107
|
-
}
|
|
108
|
-
});
|
|
109
|
-
}
|
|
110
|
-
ngOnInit() {
|
|
111
|
-
this.fxBuilderWrapperService.variables$
|
|
112
|
-
.pipe(takeUntil(this.destroy$))
|
|
113
|
-
.subscribe((variables) => {
|
|
114
|
-
console.log("Variables", variables);
|
|
115
|
-
});
|
|
116
|
-
const serviceUrl = this.fxApiService.getServiceUrl(this.setting('serviceName'));
|
|
117
|
-
this.getOptions(serviceUrl, this.setting('clinicalNotesURL'));
|
|
118
|
-
}
|
|
119
|
-
getOptions(serviceUrl, url) {
|
|
120
|
-
const finalUrl = serviceUrl + url;
|
|
121
|
-
this.http.get(finalUrl).subscribe({
|
|
122
|
-
next: (response) => {
|
|
123
|
-
// Handle API response
|
|
124
|
-
},
|
|
125
|
-
error: (err) => console.error('Error fetching options', err)
|
|
126
|
-
});
|
|
127
|
-
}
|
|
128
|
-
settings() {
|
|
129
|
-
return [
|
|
130
|
-
new FxSelectSetting({ key: 'displayMode', $title: 'Display Mode', value: 'ellipsis' }, [{ option: 'Ellipsis', value: 'ellipsis' }, { option: 'Compact', value: 'compact' }]),
|
|
131
|
-
new FxStringSetting({ key: 'placeholderLabel', $title: 'Placeholder', value: 'Select Options' }),
|
|
132
|
-
];
|
|
133
|
-
}
|
|
134
|
-
validations() {
|
|
135
|
-
return [FxValidatorService.required];
|
|
136
|
-
}
|
|
137
|
-
/** Dropdown Behavior **/
|
|
138
|
-
toggleDropdown() {
|
|
139
|
-
this.dropdownOpen = !this.dropdownOpen;
|
|
140
|
-
}
|
|
141
|
-
toggleOption(option, event) {
|
|
142
|
-
event.stopPropagation();
|
|
143
|
-
option.selected = !option.selected;
|
|
144
|
-
const index = this.findingsOptions.findIndex(o => o.value === option.value);
|
|
145
|
-
const control = this.form.get('findings').at(index);
|
|
146
|
-
if (option.selected && option.subOptions?.length) {
|
|
147
|
-
control.get('subSelection')?.setValidators(Validators.required);
|
|
148
|
-
}
|
|
149
|
-
else {
|
|
150
|
-
control.get('subSelection')?.clearValidators();
|
|
151
|
-
control.get('subSelection')?.setValue(null);
|
|
152
|
-
}
|
|
153
|
-
control.get('subSelection')?.updateValueAndValidity();
|
|
154
|
-
this.cdr.detectChanges();
|
|
155
|
-
}
|
|
156
|
-
/** Form & Label Helpers **/
|
|
157
|
-
get hasSelectedFindings() {
|
|
158
|
-
return this.findingsOptions.some(f => f.selected);
|
|
159
|
-
}
|
|
160
|
-
get selectedFindingsLabel() {
|
|
161
|
-
const selected = this.findingsOptions
|
|
162
|
-
.filter(f => {
|
|
163
|
-
if (f.selected) {
|
|
164
|
-
// If finding has sub-options → only show if a sub-option is selected
|
|
165
|
-
if (f.subOptions?.length) {
|
|
166
|
-
return !!f.subSelection;
|
|
167
|
-
}
|
|
168
|
-
// If no sub-options → always show
|
|
169
|
-
return true;
|
|
170
|
-
}
|
|
171
|
-
return false;
|
|
172
|
-
})
|
|
173
|
-
.map(f => f.label);
|
|
174
|
-
if (selected.length === 0)
|
|
175
|
-
return this.setting('placeholderLabel');
|
|
176
|
-
// Display mode logic (Compact or Ellipsis)
|
|
177
|
-
const maxCount = this.setting('displayMode') === 'compact' ? 2 : 3;
|
|
178
|
-
if (this.setting('displayMode') === 'compact') {
|
|
179
|
-
return selected.length <= maxCount
|
|
180
|
-
? selected.join(', ')
|
|
181
|
-
: `${selected.slice(0, maxCount).join(', ')} +${selected.length - maxCount} more`;
|
|
182
|
-
}
|
|
183
|
-
if (this.setting('displayMode') === 'ellipsis') {
|
|
184
|
-
return selected.length > maxCount
|
|
185
|
-
? `${selected.slice(0, maxCount).join(', ')}, ...`
|
|
186
|
-
: selected.join(', ');
|
|
187
|
-
}
|
|
188
|
-
return selected.join(', ');
|
|
189
|
-
}
|
|
190
|
-
/** Update Findings + Validation **/
|
|
191
|
-
updateFindings() {
|
|
192
|
-
const selected = this.findingsOptions
|
|
193
|
-
.filter(f => {
|
|
194
|
-
if (f.selected) {
|
|
195
|
-
// Only include in final form if:
|
|
196
|
-
// - no subOptions, or
|
|
197
|
-
// - subOptions with valid subSelection
|
|
198
|
-
if (f.subOptions?.length) {
|
|
199
|
-
return !!f.subSelection;
|
|
200
|
-
}
|
|
201
|
-
return true;
|
|
202
|
-
}
|
|
203
|
-
return false;
|
|
204
|
-
})
|
|
205
|
-
.map(f => {
|
|
206
|
-
const sub = f.subOptions?.find(s => s.value === f.subSelection) || null;
|
|
207
|
-
return {
|
|
208
|
-
label: f.label,
|
|
209
|
-
value: f.value,
|
|
210
|
-
subSelection: sub ? { label: sub.label, value: sub.value } : null
|
|
211
|
-
};
|
|
212
|
-
});
|
|
213
|
-
// Update reactive form value
|
|
214
|
-
this.form.patchValue({ findings: selected }, { emitEvent: false });
|
|
215
|
-
// Validation logic remains same
|
|
216
|
-
const invalidItems = this.findingsOptions.filter(f => f.selected && f.subOptions && !f.subSelection);
|
|
217
|
-
this.form.get('findings')?.setErrors(invalidItems.length > 0 ? { missingSubSelection: true } : null);
|
|
218
|
-
}
|
|
219
|
-
patchExistingValues(data) {
|
|
220
|
-
const formArray = this.form.get('findings');
|
|
221
|
-
formArray.clear();
|
|
222
|
-
this.findingsOptions.forEach(opt => {
|
|
223
|
-
const match = data.find(x => x.value === opt.value);
|
|
224
|
-
opt.selected = !!match;
|
|
225
|
-
opt.subSelection = match?.subSelection?.value || null;
|
|
226
|
-
if (opt.selected) {
|
|
227
|
-
formArray.push(this.fb.group({
|
|
228
|
-
value: [opt.value],
|
|
229
|
-
subSelection: [opt.subSelection]
|
|
230
|
-
}));
|
|
231
|
-
}
|
|
232
|
-
});
|
|
233
|
-
this.cdr.detectChanges();
|
|
234
|
-
this.updateFindings();
|
|
235
|
-
}
|
|
236
|
-
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 });
|
|
237
|
-
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\" class=\"relative\">\r\n <!-- Header -->\r\n <div #dropdownWrapper class=\"relative w-80\">\r\n <button 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 <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 class=\"w-5 h-5 ml-2 text-gray-500 transform transition-transform duration-200\"\r\n [class.rotate-180]=\"dropdownOpen\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\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 *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 <div *ngFor=\"let option of findingsOptions; let i = index\"\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 <!-- Checkbox + Label -->\r\n <div class=\"flex items-center gap-2\">\r\n <input type=\"checkbox\" class=\"w-4 h-4 text-blue-500 border-gray-300 rounded cursor-pointer\"\r\n [(ngModel)]=\"option.selected\" [ngModelOptions]=\"{ standalone: true }\"\r\n (click)=\"toggleOption(option, $event)\" />\r\n <label class=\"text-gray-800 font-medium cursor-pointer select-none\"\r\n (click)=\"toggleOption(option, $event)\">\r\n {{ option.label }}\r\n </label>\r\n\r\n <span *ngIf=\"option.info\" class=\"ml-auto text-blue-500 text-sm cursor-pointer\" title=\"{{ option.info }}\">\r\n \u24D8\r\n </span>\r\n </div>\r\n\r\n <!-- Radios (sub-options) -->\r\n <div class=\"flex items-center flex-wrap gap-4 ml-6 mt-2 text-sm\"\r\n *ngIf=\"option.selected && option.subOptions\">\r\n <ng-container *ngFor=\"let s of option.subOptions\">\r\n <label class=\"flex items-center space-x-1 cursor-pointer\"\r\n (click)=\"option.subSelection = s.value; option.touched = true; updateFindings()\">\r\n <input type=\"radio\" [name]=\"option.value\" [value]=\"s.value\" [(ngModel)]=\"option.subSelection\"\r\n [ngModelOptions]=\"{ standalone: true }\" class=\"text-blue-600 cursor-pointer\" />\r\n <span>{{ s.label }}</span>\r\n </label>\r\n </ng-container>\r\n\r\n <!-- Validation Message (if sub-selection is missing) -->\r\n <div *ngIf=\"option.selected && option.subOptions && !option.subSelection\"\r\n class=\"text-red-500 text-xs mt-1 w-full ml-1\">\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: FormsModule }, { 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.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: FxComponent, selector: "fx-component", inputs: ["fxData"] }] });
|
|
238
|
-
}
|
|
239
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CustomizeDropdownComponent, decorators: [{
|
|
240
|
-
type: Component,
|
|
241
|
-
args: [{ selector: 'lib-customize-dropdown', standalone: true, imports: [CommonModule, FormsModule, ReactiveFormsModule, FxComponent], template: "<fx-component [fxData]=\"fxData\" #fxComponent>\r\n <div class=\"container\">\r\n <form [formGroup]=\"form\" class=\"relative\">\r\n <!-- Header -->\r\n <div #dropdownWrapper class=\"relative w-80\">\r\n <button 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 <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 class=\"w-5 h-5 ml-2 text-gray-500 transform transition-transform duration-200\"\r\n [class.rotate-180]=\"dropdownOpen\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\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 *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 <div *ngFor=\"let option of findingsOptions; let i = index\"\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 <!-- Checkbox + Label -->\r\n <div class=\"flex items-center gap-2\">\r\n <input type=\"checkbox\" class=\"w-4 h-4 text-blue-500 border-gray-300 rounded cursor-pointer\"\r\n [(ngModel)]=\"option.selected\" [ngModelOptions]=\"{ standalone: true }\"\r\n (click)=\"toggleOption(option, $event)\" />\r\n <label class=\"text-gray-800 font-medium cursor-pointer select-none\"\r\n (click)=\"toggleOption(option, $event)\">\r\n {{ option.label }}\r\n </label>\r\n\r\n <span *ngIf=\"option.info\" class=\"ml-auto text-blue-500 text-sm cursor-pointer\" title=\"{{ option.info }}\">\r\n \u24D8\r\n </span>\r\n </div>\r\n\r\n <!-- Radios (sub-options) -->\r\n <div class=\"flex items-center flex-wrap gap-4 ml-6 mt-2 text-sm\"\r\n *ngIf=\"option.selected && option.subOptions\">\r\n <ng-container *ngFor=\"let s of option.subOptions\">\r\n <label class=\"flex items-center space-x-1 cursor-pointer\"\r\n (click)=\"option.subSelection = s.value; option.touched = true; updateFindings()\">\r\n <input type=\"radio\" [name]=\"option.value\" [value]=\"s.value\" [(ngModel)]=\"option.subSelection\"\r\n [ngModelOptions]=\"{ standalone: true }\" class=\"text-blue-600 cursor-pointer\" />\r\n <span>{{ s.label }}</span>\r\n </label>\r\n </ng-container>\r\n\r\n <!-- Validation Message (if sub-selection is missing) -->\r\n <div *ngIf=\"option.selected && option.subOptions && !option.subSelection\"\r\n class=\"text-red-500 text-xs mt-1 w-full ml-1\">\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"] }]
|
|
242
|
-
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1.HttpClient }, { type: i2.FxBuilderWrapperService }, { type: i3.ApiServiceRegistry }, { type: i4.FormBuilder }, { type: i0.ElementRef }] });
|
|
243
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"customize-dropdown.component.js","sourceRoot":"","sources":["../../../../../../projects/fx-builder-wrapper/src/lib/components/multiselect-with-form-fields/customize-dropdown.component.ts","../../../../../../projects/fx-builder-wrapper/src/lib/components/multiselect-with-form-fields/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, OnInit } from '@angular/core';\r\nimport { FormArray, FormBuilder, FormGroup, FormsModule, ReactiveFormsModule, Validators } from '@angular/forms';\r\nimport { Subject, takeUntil } from 'rxjs';\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\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,FormsModule,ReactiveFormsModule,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  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: this.fb.array([])  // Initially an empty array\r\n    });\r\n\r\n    // Initialize the form array\r\n    this.updateFindingsArray();\r\n    this.onInit.subscribe(() => this._register(this.form));\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    setTimeout(() => {\r\n      const data = [\r\n        {\r\n          \"value\": \"proclination\",\r\n          \"subSelection\": { \"label\": \"Mild\", \"value\": \"mild\" }\r\n        },\r\n      ]\r\n      this.patchExistingValues(data);\r\n    }, 2000);\r\n  }\r\n\r\n  updateFindingsArray() {\r\n    const findingsFormArray = this.form.get('findings') as FormArray;\r\n    findingsFormArray.clear();\r\n    this.findingsOptions.forEach((option) => {\r\n      if (option.selected) {\r\n        findingsFormArray.push(this.fb.group({\r\n          value: [option.value],\r\n          subSelection: [option.subSelection]\r\n        }));\r\n      }\r\n    });\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\", 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        // Handle API response\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    option.selected = !option.selected;\r\n\r\n    const index = this.findingsOptions.findIndex(o => o.value === option.value);\r\n    const control = (this.form.get('findings') as FormArray).at(index);\r\n\r\n    if (option.selected && option.subOptions?.length) {\r\n      control.get('subSelection')?.setValidators(Validators.required);\r\n    } else {\r\n      control.get('subSelection')?.clearValidators();\r\n      control.get('subSelection')?.setValue(null);\r\n    }\r\n\r\n    control.get('subSelection')?.updateValueAndValidity();\r\n    this.cdr.detectChanges();\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  patchExistingValues(data: any[]) {\r\n    const formArray = this.form.get('findings') as FormArray;\r\n    formArray.clear();\r\n\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?.subSelection?.value || null;\r\n\r\n      if (opt.selected) {\r\n        formArray.push(\r\n          this.fb.group({\r\n            value: [opt.value],\r\n            subSelection: [opt.subSelection]\r\n          })\r\n        );\r\n      }\r\n    });\r\n\r\n    this.cdr.detectChanges();\r\n    this.updateFindings();\r\n  }\r\n}\r\n","<fx-component [fxData]=\"fxData\" #fxComponent>\r\n  <div class=\"container\">\r\n    <form [formGroup]=\"form\" class=\"relative\">\r\n      <!-- Header -->\r\n      <div #dropdownWrapper class=\"relative w-80\">\r\n        <button 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          <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 class=\"w-5 h-5 ml-2 text-gray-500 transform transition-transform duration-200\"\r\n            [class.rotate-180]=\"dropdownOpen\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\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 *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          <div *ngFor=\"let option of findingsOptions; let i = index\"\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            <!-- Checkbox + Label -->\r\n            <div class=\"flex items-center gap-2\">\r\n              <input type=\"checkbox\" class=\"w-4 h-4 text-blue-500 border-gray-300 rounded cursor-pointer\"\r\n                [(ngModel)]=\"option.selected\" [ngModelOptions]=\"{ standalone: true }\"\r\n                (click)=\"toggleOption(option, $event)\" />\r\n              <label class=\"text-gray-800 font-medium cursor-pointer select-none\"\r\n                (click)=\"toggleOption(option, $event)\">\r\n                {{ option.label }}\r\n              </label>\r\n\r\n              <span *ngIf=\"option.info\" class=\"ml-auto text-blue-500 text-sm cursor-pointer\" title=\"{{ option.info }}\">\r\n                ⓘ\r\n              </span>\r\n            </div>\r\n\r\n            <!-- Radios (sub-options) -->\r\n            <div class=\"flex items-center flex-wrap gap-4 ml-6 mt-2 text-sm\"\r\n              *ngIf=\"option.selected && option.subOptions\">\r\n              <ng-container *ngFor=\"let s of option.subOptions\">\r\n                <label class=\"flex items-center space-x-1 cursor-pointer\"\r\n                  (click)=\"option.subSelection = s.value; option.touched = true; updateFindings()\">\r\n                  <input type=\"radio\" [name]=\"option.value\" [value]=\"s.value\" [(ngModel)]=\"option.subSelection\"\r\n                    [ngModelOptions]=\"{ standalone: true }\" class=\"text-blue-600 cursor-pointer\" />\r\n                  <span>{{ s.label }}</span>\r\n                </label>\r\n              </ng-container>\r\n\r\n              <!-- Validation Message (if sub-selection is missing) -->\r\n              <div *ngIf=\"option.selected && option.subOptions && !option.subSelection\"\r\n                class=\"text-red-500 text-xs mt-1 w-full ml-1\">\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"]}
|
package/esm2022/lib/components/multiselect-with-form-fields/multiselect-dropdown.component.mjs
DELETED
|
@@ -1,166 +0,0 @@
|
|
|
1
|
-
import { CommonModule } from '@angular/common';
|
|
2
|
-
import { Component, HostListener } 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 MultiselectDropdownComponentForm extends FxBaseComponent {
|
|
13
|
-
cdr;
|
|
14
|
-
http;
|
|
15
|
-
fxBuilderWrapperService;
|
|
16
|
-
fxApiService;
|
|
17
|
-
fb;
|
|
18
|
-
eRef;
|
|
19
|
-
form;
|
|
20
|
-
dropdownOpen = false;
|
|
21
|
-
searchTerm = '';
|
|
22
|
-
placeholder = 'Select items';
|
|
23
|
-
destroy$ = new Subject();
|
|
24
|
-
formObject = {};
|
|
25
|
-
options = [
|
|
26
|
-
{ label: 'Option 1', value: 'opt1', selected: false },
|
|
27
|
-
{ label: 'Option 2', value: 'opt2', selected: false },
|
|
28
|
-
{ label: 'Option 3', value: 'opt3', selected: false },
|
|
29
|
-
{ label: 'Option 4', value: 'opt4', selected: false },
|
|
30
|
-
{ label: 'Option 5', value: 'opt5', selected: false },
|
|
31
|
-
];
|
|
32
|
-
constructor(cdr, http, fxBuilderWrapperService, fxApiService, fb, eRef) {
|
|
33
|
-
super(cdr);
|
|
34
|
-
this.cdr = cdr;
|
|
35
|
-
this.http = http;
|
|
36
|
-
this.fxBuilderWrapperService = fxBuilderWrapperService;
|
|
37
|
-
this.fxApiService = fxApiService;
|
|
38
|
-
this.fb = fb;
|
|
39
|
-
this.eRef = eRef;
|
|
40
|
-
this.form = this.fb.group({
|
|
41
|
-
selectedOptionsMultiForm: [[], arrayRequiredValidator]
|
|
42
|
-
});
|
|
43
|
-
this.onInit.subscribe(() => this._register(this.form));
|
|
44
|
-
}
|
|
45
|
-
/** Close dropdown when clicking outside */
|
|
46
|
-
onClickOutside(event) {
|
|
47
|
-
if (this.dropdownOpen && !this.eRef.nativeElement.contains(event.target)) {
|
|
48
|
-
this.dropdownOpen = false;
|
|
49
|
-
this.cdr.detectChanges();
|
|
50
|
-
}
|
|
51
|
-
}
|
|
52
|
-
ngOnInit() {
|
|
53
|
-
this.fxBuilderWrapperService.variables$
|
|
54
|
-
.pipe(takeUntil(this.destroy$))
|
|
55
|
-
.subscribe((variables) => {
|
|
56
|
-
console.log("Variables");
|
|
57
|
-
// If your variables or settings change at runtime and you want to re-evaluate:
|
|
58
|
-
this.applyValidation();
|
|
59
|
-
});
|
|
60
|
-
const serviceUrl = this.fxApiService.getServiceUrl(this.setting('serviceName'));
|
|
61
|
-
this.getOptions(serviceUrl, this.setting('clinicalNotesURL'));
|
|
62
|
-
}
|
|
63
|
-
getOptions(serviceUrl, url) {
|
|
64
|
-
const finalUrl = serviceUrl + url;
|
|
65
|
-
this.http.get(finalUrl).subscribe({
|
|
66
|
-
next: (response) => {
|
|
67
|
-
// Future API logic here
|
|
68
|
-
},
|
|
69
|
-
error: (err) => console.error('Error fetching options', err)
|
|
70
|
-
});
|
|
71
|
-
}
|
|
72
|
-
toggleDropdown() {
|
|
73
|
-
this.dropdownOpen = !this.dropdownOpen;
|
|
74
|
-
if (this.dropdownOpen)
|
|
75
|
-
this.searchTerm = '';
|
|
76
|
-
}
|
|
77
|
-
/** Filter options based on search term */
|
|
78
|
-
get filteredOptions() {
|
|
79
|
-
const term = this.searchTerm.toLowerCase();
|
|
80
|
-
return this.options.filter(opt => opt.label.toLowerCase().includes(term));
|
|
81
|
-
}
|
|
82
|
-
toggleOption(option, event) {
|
|
83
|
-
event.stopPropagation();
|
|
84
|
-
option.selected = !option.selected;
|
|
85
|
-
this.updateSelectedValues();
|
|
86
|
-
}
|
|
87
|
-
/** Update reactive form with selected items */
|
|
88
|
-
updateSelectedValues() {
|
|
89
|
-
const selectedValues = this.options
|
|
90
|
-
.filter(o => o.selected)
|
|
91
|
-
.map(o => ({ label: o.label, value: o.value }));
|
|
92
|
-
this.form.patchValue({ selectedOptionsMultiForm: selectedValues }, { emitEvent: false });
|
|
93
|
-
}
|
|
94
|
-
/** Placeholder / Display logic */
|
|
95
|
-
get selectedLabel() {
|
|
96
|
-
const selected = this.options.filter(o => o.selected).map(o => o.label);
|
|
97
|
-
if (selected.length === 0)
|
|
98
|
-
return this.placeholder = this.setting('placeholderLabel') ?? this.placeholder;
|
|
99
|
-
const maxCount = this.setting('displayMode') === 'compact' ? 2 : 3;
|
|
100
|
-
if (this.setting('displayMode') === 'compact') {
|
|
101
|
-
return selected.length <= maxCount
|
|
102
|
-
? selected.join(', ')
|
|
103
|
-
: `${selected.slice(0, maxCount).join(', ')} +${selected.length - maxCount} more`;
|
|
104
|
-
}
|
|
105
|
-
if (this.setting('displayMode') === 'ellipsis') {
|
|
106
|
-
return selected.length > maxCount
|
|
107
|
-
? `${selected.slice(0, maxCount).join(', ')}, ...`
|
|
108
|
-
: selected.join(', ');
|
|
109
|
-
}
|
|
110
|
-
return selected.join(', ');
|
|
111
|
-
}
|
|
112
|
-
onSubmit() {
|
|
113
|
-
console.log('Form Value:', this.form.value);
|
|
114
|
-
}
|
|
115
|
-
get getDropdownSearch() {
|
|
116
|
-
return this.setting('dropDownSearch');
|
|
117
|
-
}
|
|
118
|
-
/**
|
|
119
|
-
* Apply or remove the array-required validator depending on the FX setting.
|
|
120
|
-
*/
|
|
121
|
-
applyValidation() {
|
|
122
|
-
const control = this.form.get('selectedOptionsMultiForm');
|
|
123
|
-
console.log(this.form);
|
|
124
|
-
const shouldRequire = this.setting('validationRequired') === 'yes';
|
|
125
|
-
if (!control)
|
|
126
|
-
return;
|
|
127
|
-
if (shouldRequire) {
|
|
128
|
-
control.setValidators(arrayRequiredValidator);
|
|
129
|
-
}
|
|
130
|
-
else {
|
|
131
|
-
control.clearValidators();
|
|
132
|
-
}
|
|
133
|
-
// Recompute validity after changing validators
|
|
134
|
-
control.updateValueAndValidity({ emitEvent: false });
|
|
135
|
-
}
|
|
136
|
-
settings() {
|
|
137
|
-
return [
|
|
138
|
-
new FxSelectSetting({ key: 'displayMode', $title: 'Display Mode', value: 'ellipsis' }, [{ option: 'Ellipsis', value: 'ellipsis' }, { option: 'Compact', value: 'compact' }]),
|
|
139
|
-
new FxSelectSetting({ key: 'validationRequired', $title: 'Validation Required', value: 'yes' }, [{ option: 'Yes', value: 'yes' }, { option: 'No', value: 'no' }]),
|
|
140
|
-
new FxSelectSetting({ key: 'dropDownSearch', $title: 'Dropdown Search', value: 'yes' }, [{ option: 'Yes', value: 'yes' }, { option: 'No', value: 'no' }]),
|
|
141
|
-
new FxStringSetting({ key: 'placeholderLabel', $title: 'Placeholder', value: 'Select Options' }),
|
|
142
|
-
new FxSelectSetting({ key: 'serviceName', $title: 'Service', value: '' }, [{ option: 'User Service', value: 'user_service' }, { option: 'Patient Service', value: 'patient_service' }, { option: 'Workflow Service', value: 'workflow_service' }]),
|
|
143
|
-
new FxStringSetting({ key: 'errorMessage', $title: 'Error Message', value: 'Please select at least one option' }),
|
|
144
|
-
];
|
|
145
|
-
}
|
|
146
|
-
validations() {
|
|
147
|
-
return [FxValidatorService.required];
|
|
148
|
-
}
|
|
149
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MultiselectDropdownComponentForm, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.HttpClient }, { token: i2.FxBuilderWrapperService }, { token: i3.ApiServiceRegistry }, { token: i4.FormBuilder }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
150
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: MultiselectDropdownComponentForm, isStandalone: true, selector: "multiselect-dropdown-form", host: { listeners: { "document:click": "onClickOutside($event)" } }, 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 <!-- Dropdown header -->\r\n <div class=\"dropdown\" #dropdownWrapper class=\"relative w-80\">\r\n <button type=\"button\" class=\"dropdown-header\" (click)=\"toggleDropdown()\">\r\n <span>{{ selectedLabel }}</span>\r\n <span class=\"arrow\" [class.open]=\"dropdownOpen\">▾</span>\r\n </button>\r\n\r\n <!-- Dropdown panel -->\r\n <div *ngIf=\"dropdownOpen\" class=\"dropdown-panel\">\r\n <!-- Search -->\r\n <div class=\"search-box\" *ngIf=\"getDropdownSearch === 'yes';\">\r\n <input\r\n type=\"text\"\r\n placeholder=\"Search...\"\r\n [(ngModel)]=\"searchTerm\"\r\n [ngModelOptions]=\"{ standalone: true }\"\r\n class=\"search-input\"\r\n />\r\n </div>\r\n\r\n <!-- Options -->\r\n <ng-container *ngIf=\"filteredOptions.length > 0; else noRecords\">\r\n <div\r\n *ngFor=\"let option of filteredOptions\"\r\n class=\"dropdown-item\"\r\n (click)=\"toggleOption(option, $event)\"\r\n >\r\n <input\r\n type=\"checkbox\"\r\n [(ngModel)]=\"option.selected\"\r\n [ngModelOptions]=\"{ standalone: true }\"\r\n class=\"checkbox\"\r\n />\r\n <label class=\"option-label\">{{ option.label }}</label>\r\n </div>\r\n </ng-container>\r\n\r\n <!-- No Records Template -->\r\n <ng-template #noRecords>\r\n <div class=\"no-records\">\r\n {{ searchTerm ? 'No records found' : 'No options available' }}\r\n </div>\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <small\r\n *ngIf=\"form.get('selectedOptionsMultiForm')?.touched && form.get('selectedOptionsMultiForm')?.errors?.['required']\"\r\n class=\"text-red-500 block mt-1\"\r\n >\r\n {{ setting('errorMessage') }}\r\n </small>\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-header:hover{border-color:#007bff}.arrow{font-size:10px;color:#555;transition:transform .2s}.arrow.open{transform:rotate(180deg)}.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}.search-box{padding:6px;border-bottom:1px solid #eee}.search-input{width:100%;padding:5px;border:1px solid #ccc;border-radius:4px;outline:none}.dropdown-item{padding:6px 8px;display:flex;align-items:center;gap:8px;cursor:pointer}.dropdown-item:hover{background-color:#f8f9fa}.checkbox{cursor:pointer}.option-label{cursor:pointer;flex-grow:1}.submit-btn{margin-top:1rem;padding:6px 12px;background:#007bff;color:#fff;border:none;border-radius:4px;cursor:pointer}.no-records{text-align:center;padding:10px;color:#888;font-size:13px;-webkit-user-select:none;user-select:none}\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.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"] }] });
|
|
151
|
-
}
|
|
152
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: MultiselectDropdownComponentForm, decorators: [{
|
|
153
|
-
type: Component,
|
|
154
|
-
args: [{ selector: 'multiselect-dropdown-form', 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 <!-- Dropdown header -->\r\n <div class=\"dropdown\" #dropdownWrapper class=\"relative w-80\">\r\n <button type=\"button\" class=\"dropdown-header\" (click)=\"toggleDropdown()\">\r\n <span>{{ selectedLabel }}</span>\r\n <span class=\"arrow\" [class.open]=\"dropdownOpen\">▾</span>\r\n </button>\r\n\r\n <!-- Dropdown panel -->\r\n <div *ngIf=\"dropdownOpen\" class=\"dropdown-panel\">\r\n <!-- Search -->\r\n <div class=\"search-box\" *ngIf=\"getDropdownSearch === 'yes';\">\r\n <input\r\n type=\"text\"\r\n placeholder=\"Search...\"\r\n [(ngModel)]=\"searchTerm\"\r\n [ngModelOptions]=\"{ standalone: true }\"\r\n class=\"search-input\"\r\n />\r\n </div>\r\n\r\n <!-- Options -->\r\n <ng-container *ngIf=\"filteredOptions.length > 0; else noRecords\">\r\n <div\r\n *ngFor=\"let option of filteredOptions\"\r\n class=\"dropdown-item\"\r\n (click)=\"toggleOption(option, $event)\"\r\n >\r\n <input\r\n type=\"checkbox\"\r\n [(ngModel)]=\"option.selected\"\r\n [ngModelOptions]=\"{ standalone: true }\"\r\n class=\"checkbox\"\r\n />\r\n <label class=\"option-label\">{{ option.label }}</label>\r\n </div>\r\n </ng-container>\r\n\r\n <!-- No Records Template -->\r\n <ng-template #noRecords>\r\n <div class=\"no-records\">\r\n {{ searchTerm ? 'No records found' : 'No options available' }}\r\n </div>\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <small\r\n *ngIf=\"form.get('selectedOptionsMultiForm')?.touched && form.get('selectedOptionsMultiForm')?.errors?.['required']\"\r\n class=\"text-red-500 block mt-1\"\r\n >\r\n {{ setting('errorMessage') }}\r\n </small>\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-header:hover{border-color:#007bff}.arrow{font-size:10px;color:#555;transition:transform .2s}.arrow.open{transform:rotate(180deg)}.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}.search-box{padding:6px;border-bottom:1px solid #eee}.search-input{width:100%;padding:5px;border:1px solid #ccc;border-radius:4px;outline:none}.dropdown-item{padding:6px 8px;display:flex;align-items:center;gap:8px;cursor:pointer}.dropdown-item:hover{background-color:#f8f9fa}.checkbox{cursor:pointer}.option-label{cursor:pointer;flex-grow:1}.submit-btn{margin-top:1rem;padding:6px 12px;background:#007bff;color:#fff;border:none;border-radius:4px;cursor:pointer}.no-records{text-align:center;padding:10px;color:#888;font-size:13px;-webkit-user-select:none;user-select:none}\n"] }]
|
|
155
|
-
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1.HttpClient }, { type: i2.FxBuilderWrapperService }, { type: i3.ApiServiceRegistry }, { type: i4.FormBuilder }, { type: i0.ElementRef }], propDecorators: { onClickOutside: [{
|
|
156
|
-
type: HostListener,
|
|
157
|
-
args: ['document:click', ['$event']]
|
|
158
|
-
}] } });
|
|
159
|
-
function arrayRequiredValidator(control) {
|
|
160
|
-
const value = control.value;
|
|
161
|
-
if (Array.isArray(value) && value.length === 0) {
|
|
162
|
-
return { required: true };
|
|
163
|
-
}
|
|
164
|
-
return null;
|
|
165
|
-
}
|
|
166
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multiselect-dropdown.component.js","sourceRoot":"","sources":["../../../../../../projects/fx-builder-wrapper/src/lib/components/multiselect-with-form-fields/multiselect-dropdown.component.ts","../../../../../../projects/fx-builder-wrapper/src/lib/components/multiselect-with-form-fields/multiselect-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { HttpClient } from '@angular/common/http';\r\nimport { ChangeDetectorRef, Component, DoCheck, ElementRef, HostListener, OnInit } from '@angular/core';\r\nimport { FormBuilder, FormGroup, FormsModule, ReactiveFormsModule, Validators, AbstractControl, ValidationErrors } from '@angular/forms';\r\nimport { ApiServiceRegistry } from '@instantsys-labs/core';\r\nimport { FxBaseComponent, FxComponent, FxSelectSetting, FxSetting, FxStringSetting, FxValidation, FxValidatorService } from '@instantsys-labs/fx';\r\nimport { Subject, takeUntil } from 'rxjs';\r\nimport { FxBuilderWrapperService } from '../../fx-builder-wrapper.service';\r\n\r\ninterface MultiSelectOption {\r\n  label: string;\r\n  value: string;\r\n  selected: boolean;\r\n}\r\n\r\n@Component({\r\n  selector: 'multiselect-dropdown-form',\r\n  standalone: true,\r\n  imports: [CommonModule, ReactiveFormsModule, FormsModule, FxComponent],\r\n  templateUrl: './multiselect-dropdown.component.html',\r\n  styleUrl: './multiselect-dropdown.component.css'\r\n})\r\nexport class MultiselectDropdownComponentForm extends FxBaseComponent implements OnInit {\r\n  form!: FormGroup;\r\n  dropdownOpen = false;\r\n  searchTerm = '';\r\n  placeholder = 'Select items';\r\n  private destroy$ = new Subject<Boolean>();\r\n  formObject: object = {};\r\n\r\n  options: MultiSelectOption[] = [\r\n    { label: 'Option 1', value: 'opt1', selected: false },\r\n    { label: 'Option 2', value: 'opt2', selected: false },\r\n    { label: 'Option 3', value: 'opt3', selected: false },\r\n    { label: 'Option 4', value: 'opt4', selected: false },\r\n    { label: 'Option 5', value: 'opt5', selected: false },\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      selectedOptionsMultiForm: [[], arrayRequiredValidator]\r\n    });\r\n    this.onInit.subscribe(() => this._register(this.form));\r\n  }\r\n\r\n  /** Close dropdown when clicking outside */\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  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        // If your variables or settings change at runtime and you want to re-evaluate:\r\n        this.applyValidation();\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  toggleDropdown() {\r\n    this.dropdownOpen = !this.dropdownOpen;\r\n    if (this.dropdownOpen) this.searchTerm = '';\r\n  }\r\n\r\n  /** Filter options based on search term */\r\n  get filteredOptions(): MultiSelectOption[] {\r\n    const term = this.searchTerm.toLowerCase();\r\n    return this.options.filter(opt => opt.label.toLowerCase().includes(term));\r\n  }\r\n\r\n  toggleOption(option: MultiSelectOption, event: Event) {\r\n    event.stopPropagation();\r\n    option.selected = !option.selected;\r\n    this.updateSelectedValues();\r\n  }\r\n\r\n  /** Update reactive form with selected items */\r\n  updateSelectedValues() {\r\n    const selectedValues = this.options\r\n      .filter(o => o.selected)\r\n      .map(o => ({ label: o.label, value: o.value }));\r\n    this.form.patchValue({ selectedOptionsMultiForm: selectedValues }, { emitEvent: false });\r\n  }\r\n\r\n  /** Placeholder / Display logic */\r\n  get selectedLabel(): string {\r\n    const selected = this.options.filter(o => o.selected).map(o => o.label);\r\n\r\n    if (selected.length === 0) return this.placeholder = this.setting('placeholderLabel') ?? this.placeholder;\r\n\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  onSubmit() {\r\n    console.log('Form Value:', this.form.value);\r\n  }\r\n\r\n  get getDropdownSearch(): string {\r\n    return this.setting('dropDownSearch');\r\n  }\r\n\r\n/**\r\n   * Apply or remove the array-required validator depending on the FX setting.\r\n   */\r\n  applyValidation() {\r\n    const control = this.form.get('selectedOptionsMultiForm');\r\n    console.log(this.form)\r\n    const shouldRequire = this.setting('validationRequired') === 'yes';\r\n\r\n    if (!control) return;\r\n\r\n    if (shouldRequire) {\r\n      control.setValidators(arrayRequiredValidator);\r\n    } else {\r\n      control.clearValidators();\r\n    }\r\n\r\n    // Recompute validity after changing validators\r\n    control.updateValueAndValidity({ emitEvent: false });\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 FxSelectSetting({ key: 'validationRequired', $title: 'Validation Required', value: 'yes' }, [{ option: 'Yes', value: 'yes' }, { option: 'No', value: 'no' }]),\r\n      new FxSelectSetting({ key: 'dropDownSearch', $title: 'Dropdown Search', value: 'yes' }, [{ option: 'Yes', value: 'yes' }, { option: 'No', value: 'no' }]),\r\n      new FxStringSetting({ key: 'placeholderLabel', $title: 'Placeholder', value: 'Select Options' }),\r\n      new FxSelectSetting({ key: 'serviceName', $title: 'Service', value: '' }, [{ option: 'User Service', value: 'user_service' }, { option: 'Patient Service', value: 'patient_service' }, { option: 'Workflow Service', value: 'workflow_service' }]),\r\n      new FxStringSetting({ key: 'errorMessage', $title: 'Error Message', value: 'Please select at least one option' }),\r\n    ];\r\n  }\r\n\r\n  protected validations(): FxValidation[] {\r\n    return [FxValidatorService.required];\r\n  }\r\n}\r\n\r\nfunction arrayRequiredValidator(control: AbstractControl): ValidationErrors | null {\r\n  const value = control.value;\r\n  if (Array.isArray(value) && value.length === 0) {\r\n    return { required: true };\r\n  }\r\n  return null;\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            <!-- Dropdown header -->\r\n            <div class=\"dropdown\" #dropdownWrapper class=\"relative w-80\">\r\n                <button type=\"button\" class=\"dropdown-header\" (click)=\"toggleDropdown()\">\r\n                    <span>{{ selectedLabel }}</span>\r\n                    <span class=\"arrow\" [class.open]=\"dropdownOpen\">&#9662;</span>\r\n                </button>\r\n\r\n                <!-- Dropdown panel -->\r\n                <div *ngIf=\"dropdownOpen\" class=\"dropdown-panel\">\r\n                    <!-- Search -->\r\n                    <div class=\"search-box\" *ngIf=\"getDropdownSearch === 'yes';\">\r\n                        <input\r\n                            type=\"text\"\r\n                            placeholder=\"Search...\"\r\n                            [(ngModel)]=\"searchTerm\"\r\n                            [ngModelOptions]=\"{ standalone: true }\"\r\n                            class=\"search-input\"\r\n                        />\r\n                    </div>\r\n\r\n                    <!-- Options -->\r\n                    <ng-container *ngIf=\"filteredOptions.length > 0; else noRecords\">\r\n                        <div\r\n                            *ngFor=\"let option of filteredOptions\"\r\n                            class=\"dropdown-item\"\r\n                            (click)=\"toggleOption(option, $event)\"\r\n                        >\r\n                            <input\r\n                                type=\"checkbox\"\r\n                                [(ngModel)]=\"option.selected\"\r\n                                [ngModelOptions]=\"{ standalone: true }\"\r\n                                class=\"checkbox\"\r\n                            />\r\n                            <label class=\"option-label\">{{ option.label }}</label>\r\n                        </div>\r\n                    </ng-container>\r\n\r\n                    <!-- No Records Template -->\r\n                    <ng-template #noRecords>\r\n                        <div class=\"no-records\">\r\n                            {{ searchTerm ? 'No records found' : 'No options available' }}\r\n                        </div>\r\n                    </ng-template>\r\n                </div>\r\n            </div>\r\n\r\n            <small\r\n                *ngIf=\"form.get('selectedOptionsMultiForm')?.touched && form.get('selectedOptionsMultiForm')?.errors?.['required']\"\r\n                class=\"text-red-500 block mt-1\"\r\n                >\r\n                {{ setting('errorMessage') }}\r\n            </small>\r\n        </form>\r\n    </div>\r\n</fx-component>\r\n"]}
|
|
@@ -1,60 +0,0 @@
|
|
|
1
|
-
import { HttpClient } from '@angular/common/http';
|
|
2
|
-
import { AfterViewInit, ChangeDetectorRef, ElementRef, OnInit } from '@angular/core';
|
|
3
|
-
import { FormBuilder, FormGroup } from '@angular/forms';
|
|
4
|
-
import { ApiServiceRegistry } from '@instantsys-labs/core';
|
|
5
|
-
import { FxBaseComponent, FxSetting, FxValidation } from '@instantsys-labs/fx';
|
|
6
|
-
import { FxBuilderWrapperService } from '../../fx-builder-wrapper.service';
|
|
7
|
-
import * as i0 from "@angular/core";
|
|
8
|
-
interface SubOption {
|
|
9
|
-
label: string;
|
|
10
|
-
value: string;
|
|
11
|
-
}
|
|
12
|
-
interface CustomDropdownOption {
|
|
13
|
-
label: string;
|
|
14
|
-
value: string;
|
|
15
|
-
info?: string;
|
|
16
|
-
selected: boolean;
|
|
17
|
-
subOptions?: SubOption[];
|
|
18
|
-
subSelection?: string | null;
|
|
19
|
-
disabled?: boolean;
|
|
20
|
-
touched?: boolean;
|
|
21
|
-
}
|
|
22
|
-
interface CustomDropdownConfig {
|
|
23
|
-
displayMode?: 'compact' | 'ellipsis';
|
|
24
|
-
placeholderLabel?: string;
|
|
25
|
-
}
|
|
26
|
-
export declare class CustomizeDropdownComponent extends FxBaseComponent implements OnInit, AfterViewInit {
|
|
27
|
-
private cdr;
|
|
28
|
-
private http;
|
|
29
|
-
private fxBuilderWrapperService;
|
|
30
|
-
private fxApiService;
|
|
31
|
-
private fb;
|
|
32
|
-
private eRef;
|
|
33
|
-
private destroy$;
|
|
34
|
-
form: FormGroup;
|
|
35
|
-
formObject: object;
|
|
36
|
-
dropdownOpen: boolean;
|
|
37
|
-
formSubmitted: boolean;
|
|
38
|
-
findingsOptions: CustomDropdownOption[];
|
|
39
|
-
config: CustomDropdownConfig;
|
|
40
|
-
constructor(cdr: ChangeDetectorRef, http: HttpClient, fxBuilderWrapperService: FxBuilderWrapperService, fxApiService: ApiServiceRegistry, fb: FormBuilder, eRef: ElementRef);
|
|
41
|
-
ngAfterViewInit(): void;
|
|
42
|
-
ngOnInit(): void;
|
|
43
|
-
getOptions(serviceUrl: string, url: string): void;
|
|
44
|
-
protected settings(): FxSetting[];
|
|
45
|
-
protected validations(): FxValidation[];
|
|
46
|
-
/** Dropdown Behavior **/
|
|
47
|
-
toggleDropdown(): void;
|
|
48
|
-
toggleOption(option: CustomDropdownOption, event: Event): void;
|
|
49
|
-
/** Form & Label Helpers **/
|
|
50
|
-
get hasSelectedFindings(): boolean;
|
|
51
|
-
get selectedFindingsLabel(): string;
|
|
52
|
-
/** Update Findings + Validation **/
|
|
53
|
-
updateFindings(): void;
|
|
54
|
-
onSubmit(): void;
|
|
55
|
-
patchExistingValues(data: any[]): void;
|
|
56
|
-
selectSubOption(option: any, subOption: any): void;
|
|
57
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<CustomizeDropdownComponent, never>;
|
|
58
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<CustomizeDropdownComponent, "lib-customize-dropdown", never, {}, {}, never, never, true, never>;
|
|
59
|
-
}
|
|
60
|
-
export {};
|