tango-app-ui-manage-tickets 3.7.0-beta.49 → 3.7.0-beta.50
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/ticket-filter-panel/ticket-filter-panel.component.mjs +248 -31
- package/esm2022/lib/components/ticket-footfall-new/ticket-footfall-new.component.mjs +81 -26
- package/esm2022/lib/services/ticket.service.mjs +45 -4
- package/fesm2022/tango-app-ui-manage-tickets.mjs +368 -56
- package/fesm2022/tango-app-ui-manage-tickets.mjs.map +1 -1
- package/lib/components/ticket-filter-panel/ticket-filter-panel.component.d.ts +41 -6
- package/lib/components/ticket-footfall-new/ticket-footfall-new.component.d.ts +6 -1
- package/lib/services/ticket.service.d.ts +16 -2
- package/package.json +1 -1
|
@@ -1,12 +1,17 @@
|
|
|
1
|
-
import { Component, EventEmitter, HostListener, Output } from '@angular/core';
|
|
1
|
+
import { Component, EventEmitter, HostListener, Input, Output } from '@angular/core';
|
|
2
2
|
import * as i0 from "@angular/core";
|
|
3
3
|
import * as i1 from "@angular/forms";
|
|
4
|
-
import * as i2 from "
|
|
4
|
+
import * as i2 from "../../services/ticket.service";
|
|
5
|
+
import * as i3 from "@angular/common";
|
|
5
6
|
export class TicketFilterPanelComponent {
|
|
6
7
|
fb;
|
|
7
8
|
eRef;
|
|
9
|
+
service;
|
|
8
10
|
apply = new EventEmitter();
|
|
9
11
|
panelClosed = new EventEmitter(); // ✅ not `close`
|
|
12
|
+
permissionType = null;
|
|
13
|
+
userType = null; // e.g. 'tango' or others
|
|
14
|
+
client;
|
|
10
15
|
onDocumentClick(event) {
|
|
11
16
|
if (!this.isOpen)
|
|
12
17
|
return;
|
|
@@ -33,6 +38,7 @@ export class TicketFilterPanelComponent {
|
|
|
33
38
|
'Under tango review',
|
|
34
39
|
'Tango review done',
|
|
35
40
|
'Expired',
|
|
41
|
+
'Tango Expired',
|
|
36
42
|
];
|
|
37
43
|
conditionOptions = [
|
|
38
44
|
{ value: 'gt', label: '> Greater than' },
|
|
@@ -42,13 +48,14 @@ export class TicketFilterPanelComponent {
|
|
|
42
48
|
{ value: 'between', label: 'Between' },
|
|
43
49
|
];
|
|
44
50
|
filterForm;
|
|
45
|
-
constructor(fb, eRef) {
|
|
51
|
+
constructor(fb, eRef, service) {
|
|
46
52
|
this.fb = fb;
|
|
47
53
|
this.eRef = eRef;
|
|
54
|
+
this.service = service;
|
|
48
55
|
this.filterForm = this.fb.group({
|
|
49
|
-
status: [
|
|
50
|
-
reviewerCondition: [
|
|
51
|
-
reviewerFrom: [
|
|
56
|
+
status: [[]],
|
|
57
|
+
reviewerCondition: [null],
|
|
58
|
+
reviewerFrom: [null],
|
|
52
59
|
reviewerTo: [null],
|
|
53
60
|
approverCondition: [null],
|
|
54
61
|
approverFrom: [null],
|
|
@@ -56,15 +63,206 @@ export class TicketFilterPanelComponent {
|
|
|
56
63
|
tangoCondition: [null],
|
|
57
64
|
tangoFrom: [null],
|
|
58
65
|
tangoTo: [null],
|
|
59
|
-
|
|
66
|
+
reviewedBy: [[]],
|
|
67
|
+
approvedBy: [[]],
|
|
60
68
|
});
|
|
61
69
|
}
|
|
70
|
+
get isTango() {
|
|
71
|
+
return this.userType === 'tango';
|
|
72
|
+
}
|
|
73
|
+
// SHOW / HIDE RULES
|
|
74
|
+
// Reviewer accuracy and "Reviewed by" visible for tango + reviewer
|
|
75
|
+
get showReviewerAccuracy() {
|
|
76
|
+
return this.isTango || this.permissionType === 'review';
|
|
77
|
+
}
|
|
78
|
+
get showReviewedBy() {
|
|
79
|
+
return this.isTango || this.permissionType === 'review';
|
|
80
|
+
}
|
|
81
|
+
// Approver accuracy and "Approved by" visible for tango + approver
|
|
82
|
+
get showApproverAccuracy() {
|
|
83
|
+
return this.isTango || this.permissionType === 'approve';
|
|
84
|
+
}
|
|
85
|
+
get showApprovedBy() {
|
|
86
|
+
return this.isTango || this.permissionType === 'approve';
|
|
87
|
+
}
|
|
88
|
+
// Tango accuracy only for tango users
|
|
89
|
+
get showTangoAccuracy() {
|
|
90
|
+
return this.isTango;
|
|
91
|
+
}
|
|
92
|
+
statusDropdownOpen = false;
|
|
93
|
+
isStatusSelected(status) {
|
|
94
|
+
const selected = this.filterForm.value.status || [];
|
|
95
|
+
return selected.includes(status);
|
|
96
|
+
}
|
|
97
|
+
onStatusChange(event, status) {
|
|
98
|
+
const checkbox = event.target;
|
|
99
|
+
const selected = this.filterForm.value.status || [];
|
|
100
|
+
let next;
|
|
101
|
+
if (checkbox.checked) {
|
|
102
|
+
next = selected.includes(status) ? selected : [...selected, status];
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
next = selected.filter(s => s !== status);
|
|
106
|
+
}
|
|
107
|
+
this.filterForm.patchValue({ status: next });
|
|
108
|
+
}
|
|
109
|
+
toggleStatusDropdown(event) {
|
|
110
|
+
if (event) {
|
|
111
|
+
event.stopPropagation(); // avoid document click closing it
|
|
112
|
+
}
|
|
113
|
+
this.statusDropdownOpen = !this.statusDropdownOpen;
|
|
114
|
+
}
|
|
115
|
+
// Are all statuses currently selected?
|
|
116
|
+
areAllStatusesSelected() {
|
|
117
|
+
const selected = this.filterForm.value.status || [];
|
|
118
|
+
return selected.length > 0 && selected.length === this.statusOptions.length;
|
|
119
|
+
}
|
|
120
|
+
// Handle "Select All" checkbox change
|
|
121
|
+
onStatusSelectAllChange(event) {
|
|
122
|
+
const checkbox = event.target;
|
|
123
|
+
if (checkbox.checked) {
|
|
124
|
+
// Select every status
|
|
125
|
+
this.filterForm.patchValue({ status: [...this.statusOptions] });
|
|
126
|
+
}
|
|
127
|
+
else {
|
|
128
|
+
// Clear all
|
|
129
|
+
this.filterForm.patchValue({ status: [] });
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
get selectedStatuses() {
|
|
133
|
+
return this.filterForm.value.status || [];
|
|
134
|
+
}
|
|
62
135
|
isBetween(controlName) {
|
|
63
136
|
return this.filterForm.get(controlName)?.value === 'between';
|
|
64
137
|
}
|
|
138
|
+
ngOnChanges(changes) {
|
|
139
|
+
if (changes['client'] || changes['permissionType'] || changes['userType']) {
|
|
140
|
+
this.loadReviewerList();
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
userList;
|
|
144
|
+
loadReviewerList() {
|
|
145
|
+
if (!this.client) {
|
|
146
|
+
this.userList = [];
|
|
147
|
+
return;
|
|
148
|
+
}
|
|
149
|
+
let type;
|
|
150
|
+
if (this.userType === 'tango') {
|
|
151
|
+
type = 'tango'; // or 'all'
|
|
152
|
+
}
|
|
153
|
+
else if (this.permissionType === 'review') {
|
|
154
|
+
type = 'review';
|
|
155
|
+
}
|
|
156
|
+
else if (this.permissionType === 'approve') {
|
|
157
|
+
type = 'approve';
|
|
158
|
+
}
|
|
159
|
+
else {
|
|
160
|
+
this.userList = [];
|
|
161
|
+
return;
|
|
162
|
+
}
|
|
163
|
+
this.service.getReviewerApi(this.client, type).subscribe({
|
|
164
|
+
next: (res) => {
|
|
165
|
+
if (res && res.code === 200) {
|
|
166
|
+
this.userList = res.data;
|
|
167
|
+
}
|
|
168
|
+
else {
|
|
169
|
+
this.userList = [];
|
|
170
|
+
}
|
|
171
|
+
},
|
|
172
|
+
error: () => {
|
|
173
|
+
this.userList = [];
|
|
174
|
+
}
|
|
175
|
+
});
|
|
176
|
+
}
|
|
177
|
+
// open flags
|
|
178
|
+
reviewerDropdownOpen = false;
|
|
179
|
+
approverDropdownOpen = false;
|
|
180
|
+
// convenience getters
|
|
181
|
+
get selectedReviewedBy() {
|
|
182
|
+
return this.filterForm.value.reviewedBy || [];
|
|
183
|
+
}
|
|
184
|
+
get selectedApprovedBy() {
|
|
185
|
+
return this.filterForm.value.approvedBy || [];
|
|
186
|
+
}
|
|
187
|
+
// --- Reviewed By helpers ---
|
|
188
|
+
toggleReviewerDropdown(event) {
|
|
189
|
+
if (event)
|
|
190
|
+
event.stopPropagation();
|
|
191
|
+
this.reviewerDropdownOpen = !this.reviewerDropdownOpen;
|
|
192
|
+
}
|
|
193
|
+
isReviewerSelected(email) {
|
|
194
|
+
return this.selectedReviewedBy.includes(email);
|
|
195
|
+
}
|
|
196
|
+
onReviewerChange(event, email) {
|
|
197
|
+
const checkbox = event.target;
|
|
198
|
+
const selected = [...this.selectedReviewedBy];
|
|
199
|
+
if (checkbox.checked) {
|
|
200
|
+
if (!selected.includes(email))
|
|
201
|
+
selected.push(email);
|
|
202
|
+
}
|
|
203
|
+
else {
|
|
204
|
+
const i = selected.indexOf(email);
|
|
205
|
+
if (i >= 0)
|
|
206
|
+
selected.splice(i, 1);
|
|
207
|
+
}
|
|
208
|
+
this.filterForm.patchValue({ reviewedBy: selected });
|
|
209
|
+
}
|
|
210
|
+
areAllReviewersSelected() {
|
|
211
|
+
return (this.userList.length > 0 &&
|
|
212
|
+
this.selectedReviewedBy.length === this.userList.length);
|
|
213
|
+
}
|
|
214
|
+
onReviewerSelectAll(event) {
|
|
215
|
+
const checkbox = event.target;
|
|
216
|
+
if (checkbox.checked) {
|
|
217
|
+
this.filterForm.patchValue({
|
|
218
|
+
reviewedBy: this.userList.map((u) => u.email),
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
else {
|
|
222
|
+
this.filterForm.patchValue({ reviewedBy: [] });
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
// --- Approved By helpers ---
|
|
226
|
+
toggleApproverDropdown(event) {
|
|
227
|
+
if (event)
|
|
228
|
+
event.stopPropagation();
|
|
229
|
+
this.approverDropdownOpen = !this.approverDropdownOpen;
|
|
230
|
+
}
|
|
231
|
+
isApproverSelected(email) {
|
|
232
|
+
return this.selectedApprovedBy.includes(email);
|
|
233
|
+
}
|
|
234
|
+
onApproverChange(event, email) {
|
|
235
|
+
const checkbox = event.target;
|
|
236
|
+
const selected = [...this.selectedApprovedBy];
|
|
237
|
+
if (checkbox.checked) {
|
|
238
|
+
if (!selected.includes(email))
|
|
239
|
+
selected.push(email);
|
|
240
|
+
}
|
|
241
|
+
else {
|
|
242
|
+
const i = selected.indexOf(email);
|
|
243
|
+
if (i >= 0)
|
|
244
|
+
selected.splice(i, 1);
|
|
245
|
+
}
|
|
246
|
+
this.filterForm.patchValue({ approvedBy: selected });
|
|
247
|
+
}
|
|
248
|
+
areAllApproversSelected() {
|
|
249
|
+
return (this.userList.length > 0 &&
|
|
250
|
+
this.selectedApprovedBy.length === this.userList.length);
|
|
251
|
+
}
|
|
252
|
+
onApproverSelectAll(event) {
|
|
253
|
+
const checkbox = event.target;
|
|
254
|
+
if (checkbox.checked) {
|
|
255
|
+
this.filterForm.patchValue({
|
|
256
|
+
approvedBy: this.userList.map((u) => u.email),
|
|
257
|
+
});
|
|
258
|
+
}
|
|
259
|
+
else {
|
|
260
|
+
this.filterForm.patchValue({ approvedBy: [] });
|
|
261
|
+
}
|
|
262
|
+
}
|
|
65
263
|
onReset() {
|
|
66
264
|
this.filterForm.reset({
|
|
67
|
-
status:
|
|
265
|
+
status: [],
|
|
68
266
|
reviewerCondition: null,
|
|
69
267
|
reviewerFrom: null,
|
|
70
268
|
reviewerTo: null,
|
|
@@ -74,47 +272,66 @@ export class TicketFilterPanelComponent {
|
|
|
74
272
|
tangoCondition: null,
|
|
75
273
|
tangoFrom: null,
|
|
76
274
|
tangoTo: null,
|
|
77
|
-
|
|
275
|
+
reviewedBy: [],
|
|
276
|
+
approvedBy: [],
|
|
78
277
|
});
|
|
79
278
|
this.close();
|
|
80
279
|
}
|
|
81
|
-
onApply() {
|
|
82
|
-
const raw = this.filterForm.value;
|
|
83
|
-
// build a clean DTO for API / table filter
|
|
84
|
-
const payload = {
|
|
85
|
-
filterByStatus: raw.status || null,
|
|
86
|
-
filterByReviewer: this.mapAccuracy(raw.reviewerCondition, raw.reviewerFrom, raw.reviewerTo),
|
|
87
|
-
filterByApprover: this.mapAccuracy(raw.approverCondition, raw.approverFrom, raw.approverTo),
|
|
88
|
-
filterByTango: this.mapAccuracy(raw.tangoCondition, raw.tangoFrom, raw.tangoTo),
|
|
89
|
-
filterByReviewedBy: raw.reviewedBy || null,
|
|
90
|
-
fileterByApprovedBy: raw.approvedBy || null,
|
|
91
|
-
};
|
|
92
|
-
this.apply.emit(payload);
|
|
93
|
-
this.close();
|
|
94
|
-
this.showPanel = false;
|
|
95
|
-
}
|
|
96
280
|
mapAccuracy(condition, from, to) {
|
|
281
|
+
// nothing selected
|
|
97
282
|
if (!condition || from == null)
|
|
98
283
|
return null;
|
|
284
|
+
// "between" → "90 to 100"
|
|
99
285
|
if (condition === 'between') {
|
|
100
286
|
if (to == null)
|
|
101
287
|
return null;
|
|
102
|
-
return {
|
|
288
|
+
return `${from} to ${to}`;
|
|
103
289
|
}
|
|
104
|
-
|
|
290
|
+
// other conditions (gt, lt, gte, lte)
|
|
291
|
+
const opMap = {
|
|
292
|
+
gt: '>',
|
|
293
|
+
lt: '<',
|
|
294
|
+
gte: '>=',
|
|
295
|
+
lte: '<=',
|
|
296
|
+
between: '', // already handled above
|
|
297
|
+
};
|
|
298
|
+
return `${opMap[condition]} ${from}`; // e.g. ">= 90"
|
|
299
|
+
}
|
|
300
|
+
onApply() {
|
|
301
|
+
const raw = this.filterForm.value;
|
|
302
|
+
const reviewerStr = this.mapAccuracy(raw.reviewerCondition, raw.reviewerFrom, raw.reviewerTo);
|
|
303
|
+
const approverStr = this.mapAccuracy(raw.approverCondition, raw.approverFrom, raw.approverTo);
|
|
304
|
+
const tangoStr = this.mapAccuracy(raw.tangoCondition, raw.tangoFrom, raw.tangoTo);
|
|
305
|
+
const payload = {
|
|
306
|
+
filterByStatus: raw.status || [],
|
|
307
|
+
filterByReviewer: reviewerStr,
|
|
308
|
+
filterByApprover: approverStr,
|
|
309
|
+
filterByTango: tangoStr,
|
|
310
|
+
filterByReviewedBy: raw.reviewedBy || [],
|
|
311
|
+
fileterByApprovedBy: raw.approvedBy || [],
|
|
312
|
+
};
|
|
313
|
+
this.apply.emit(payload);
|
|
314
|
+
this.close();
|
|
315
|
+
this.showPanel = false;
|
|
105
316
|
}
|
|
106
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TicketFilterPanelComponent, deps: [{ token: i1.FormBuilder }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
107
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TicketFilterPanelComponent, selector: "lib-ticket-filter-panel", outputs: { apply: "apply", panelClosed: "panelClosed" }, host: { listeners: { "document:click": "onDocumentClick($event)" } }, ngImport: i0, template: "<!-- ticket-filter-panel.component.html -->\r\n<div class=\"filter-wrapper\" *ngIf=\"isOpen\">\r\n <div class=\"filter-card\">\r\n <div class=\"filter-header d-flex justify-content-between align-items-center mb-3\">\r\n <div class=\"title\">Filter Options</div>\r\n <!-- <button type=\"button\" class=\"btn-close\" (click)=\"close()\">\u2715</button> -->\r\n </div>\r\n\r\n <form [formGroup]=\"filterForm\">\r\n <!-- Status -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Status</label>\r\n <select formControlName=\"status\" class=\"form-select\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let s of statusOptions\" [ngValue]=\"s\">{{ s }}</option>\r\n </select>\r\n </div>\r\n\r\n <!-- Reviewer accuracy -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Reviewer accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-50\" formControlName=\"reviewerCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <!-- one or two inputs -->\r\n <ng-container *ngIf=\"isBetween('reviewerCondition'); else singleReviewer\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleReviewer>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Approver accuracy -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Approver accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-50\" formControlName=\"approverCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <ng-container *ngIf=\"isBetween('approverCondition'); else singleApprover\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleApprover>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Tango accuracy -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Tango accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-50\" formControlName=\"tangoCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <ng-container *ngIf=\"isBetween('tangoCondition'); else singleTango\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleTango>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Approved By -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Approved by</label>\r\n <input\r\n class=\"form-control\"\r\n formControlName=\"approvedBy\"\r\n placeholder=\"Select / search by email\" />\r\n </div>\r\n\r\n <!-- Footer buttons -->\r\n <div class=\"d-flex justify-content-between w-100 mt-4\">\r\n <button type=\"button\" class=\"btn btn-outline w-50 me-1\" (click)=\"onReset()\">\r\n Reset\r\n </button>\r\n <button type=\"button\" class=\"btn btn-primary w-50 ms-1\" (click)=\"onApply()\">\r\n Apply\r\n </button>\r\n </div>\r\n </form>\r\n </div>\r\n</div>\r\n", styles: [".filter-wrapper{position:absolute;top:0;right:0;padding:16px;z-index:1050}.filter-card{width:350px;background:#fff;border-radius:12px;padding:16px 18px;box-shadow:0 8px 30px #00000014;font-size:13px}.filter-header .title{font-weight:600;font-size:14px}.btn-close{border:none;background:transparent;font-size:16px;cursor:pointer}.form-label{font-size:12px;margin-bottom:4px}.form-select,.form-control{font-size:13px}.gap-2{gap:4px}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "directive", type: i1.MaxValidator, selector: "input[type=number][max][formControlName],input[type=number][max][formControl],input[type=number][max][ngModel]", inputs: ["max"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }] });
|
|
317
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TicketFilterPanelComponent, deps: [{ token: i1.FormBuilder }, { token: i0.ElementRef }, { token: i2.TicketService }], target: i0.ɵɵFactoryTarget.Component });
|
|
318
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TicketFilterPanelComponent, selector: "lib-ticket-filter-panel", inputs: { permissionType: "permissionType", userType: "userType", client: "client" }, outputs: { apply: "apply", panelClosed: "panelClosed" }, host: { listeners: { "document:click": "onDocumentClick($event)" } }, usesOnChanges: true, ngImport: i0, template: "<!-- ticket-filter-panel.component.html -->\r\n<div class=\"filter-wrapper\" *ngIf=\"isOpen\">\r\n <div class=\"filter-card\">\r\n <div class=\"filter-header d-flex justify-content-between align-items-center mb-3\">\r\n <div class=\"title\">Filter Options</div>\r\n <!-- <button type=\"button\" class=\"btn-close\" (click)=\"close()\">\u2715</button> -->\r\n </div>\r\n\r\n <form [formGroup]=\"filterForm\">\r\n <!-- Status -->\r\n<div class=\"mb-3 position-relative\" (click)=\"$event.stopPropagation()\">\r\n <label class=\"form-label\">Status</label>\r\n\r\n <!-- Header / trigger -->\r\n <div\r\n class=\"status-select border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n (click)=\"toggleStatusDropdown($event)\"\r\n >\r\n <span *ngIf=\"!selectedStatuses.length\">Select</span>\r\n\r\n <span *ngIf=\"selectedStatuses.length === 1\">\r\n {{ selectedStatuses[0] }}\r\n </span>\r\n\r\n <span *ngIf=\"selectedStatuses.length > 1\" class=\"d-flex align-items-center\">\r\n <span>{{ selectedStatuses[0] }}</span>\r\n <span class=\"badge ms-2\">\r\n +{{ selectedStatuses.length - 1 }}\r\n </span>\r\n </span>\r\n\r\n </div>\r\n\r\n <!-- DROPDOWN PANEL (OVERLAY) -->\r\n <div\r\n class=\"status-dropdown position-absolute w-100 mt-1 border rounded p-1 bg-white\"\r\n style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n *ngIf=\"statusDropdownOpen\"\r\n (click)=\"$event.stopPropagation()\"\r\n >\r\n <!-- Select All -->\r\n <div class=\"form-check px-0 d-flex align-items-center mb-2\">\r\n <input\r\n type=\"checkbox\"\r\n class=\"form-check-input text-dark\"\r\n id=\"status-select-all\"\r\n [checked]=\"areAllStatusesSelected()\"\r\n (change)=\"onStatusSelectAllChange($event)\"\r\n />\r\n <label class=\"form-check-label ms-2\" for=\"status-select-all\">\r\n Select All\r\n </label>\r\n </div>\r\n\r\n\r\n <!-- Options -->\r\n <div\r\n class=\"form-check px-0 d-flex align-items-center mb-1\"\r\n *ngFor=\"let s of statusOptions\"\r\n >\r\n <input\r\n type=\"checkbox\"\r\n class=\"form-check-input text-dark\"\r\n [id]=\"'status-' + s\"\r\n [checked]=\"isStatusSelected(s)\"\r\n (change)=\"onStatusChange($event, s)\"\r\n />\r\n <label class=\"form-check-label ms-2\" [for]=\"'status-' + s\">\r\n {{ s }}\r\n </label>\r\n </div>\r\n </div>\r\n</div>\r\n\r\n\r\n\r\n <!-- Reviewer accuracy -->\r\n <div class=\"mb-3\" *ngIf=\"showReviewerAccuracy\">\r\n <label class=\"form-label\">Reviewer accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-100\" formControlName=\"reviewerCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <!-- one or two inputs -->\r\n <ng-container *ngIf=\"isBetween('reviewerCondition'); else singleReviewer\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleReviewer>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Approver accuracy -->\r\n <div class=\"mb-3\" *ngIf=\"showApproverAccuracy\">\r\n <label class=\"form-label\">Approver accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-100\" formControlName=\"approverCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <ng-container *ngIf=\"isBetween('approverCondition'); else singleApprover\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleApprover>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Tango accuracy -->\r\n <div class=\"mb-3\" *ngIf=\"showTangoAccuracy\">\r\n <label class=\"form-label\">Tango accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-50\" formControlName=\"tangoCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <ng-container *ngIf=\"isBetween('tangoCondition'); else singleTango\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleTango>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n<div\r\n class=\"mb-3 position-relative\"\r\n *ngIf=\"showReviewedBy\"\r\n (click)=\"$event.stopPropagation()\"\r\n>\r\n <label class=\"form-label\">Reviewed by</label>\r\n\r\n <!-- Trigger / display -->\r\n <div\r\n class=\"border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n (click)=\"toggleReviewerDropdown($event)\"\r\n >\r\n <!-- 0 selected -->\r\n <span *ngIf=\"!selectedReviewedBy.length\">\r\n Select\r\n </span>\r\n\r\n <!-- 1 selected -->\r\n <span *ngIf=\"selectedReviewedBy.length === 1\">\r\n {{ selectedReviewedBy[0] }}\r\n </span>\r\n\r\n <!-- >1 selected -->\r\n <span\r\n *ngIf=\"selectedReviewedBy.length > 1\"\r\n class=\"d-flex align-items-center\"\r\n >\r\n <span>{{ selectedReviewedBy[0] }}</span>\r\n <span class=\"badge ms-2\">\r\n +{{ selectedReviewedBy.length - 1 }}\r\n </span>\r\n </span>\r\n </div>\r\n\r\n <!-- Dropdown panel -->\r\n <div\r\n *ngIf=\"reviewerDropdownOpen\"\r\n class=\"position-absolute w-100 border rounded bg-white p-1 mt-1\"\r\n style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n (click)=\"$event.stopPropagation()\"\r\n >\r\n <!-- Select All -->\r\n <div class=\"form-check px-0 mb-2 d-flex align-items-center\">\r\n <input\r\n type=\"checkbox\"\r\n class=\"form-check-input\"\r\n id=\"reviewer-all\"\r\n [checked]=\"areAllReviewersSelected()\"\r\n (change)=\"onReviewerSelectAll($event)\"\r\n />\r\n <label for=\"reviewer-all\" class=\"form-check-label ms-2\">\r\n Select All\r\n </label>\r\n </div>\r\n\r\n <!-- Options -->\r\n <div\r\n class=\"form-check mb-1 px-0 d-flex align-items-center\"\r\n *ngFor=\"let u of userList\"\r\n >\r\n <input\r\n class=\"form-check-input\"\r\n type=\"checkbox\"\r\n [id]=\"'rev-' + u.email\"\r\n [checked]=\"isReviewerSelected(u.email)\"\r\n (change)=\"onReviewerChange($event, u.email)\"\r\n />\r\n <label class=\"form-check-label ms-2\" [for]=\"'rev-' + u.email\">\r\n {{ u.email }}\r\n </label>\r\n </div>\r\n </div>\r\n</div>\r\n\r\n\r\n <!-- Approved By -->\r\n<div\r\n class=\"mb-3 position-relative\"\r\n *ngIf=\"showApprovedBy\"\r\n (click)=\"$event.stopPropagation()\"\r\n>\r\n <label class=\"form-label\">Approved by</label>\r\n\r\n <!-- Trigger / display -->\r\n <div\r\n class=\"border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n (click)=\"toggleApproverDropdown($event)\"\r\n >\r\n <!-- 0 selected -->\r\n <span *ngIf=\"!selectedApprovedBy.length\">\r\n Select\r\n </span>\r\n\r\n <!-- 1 selected -->\r\n <span *ngIf=\"selectedApprovedBy.length === 1\">\r\n {{ selectedApprovedBy[0] }}\r\n </span>\r\n\r\n <!-- >1 selected -->\r\n <span\r\n *ngIf=\"selectedApprovedBy.length > 1\"\r\n class=\"d-flex align-items-center\"\r\n >\r\n <span>{{ selectedApprovedBy[0] }}</span>\r\n <span class=\"badge ms-2\">\r\n +{{ selectedApprovedBy.length - 1 }}\r\n </span>\r\n </span>\r\n\r\n </div>\r\n\r\n <!-- Dropdown panel -->\r\n <div\r\n *ngIf=\"approverDropdownOpen\"\r\n class=\"position-absolute w-100 border rounded bg-white p-2 mt-1\"\r\n style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n (click)=\"$event.stopPropagation()\"\r\n >\r\n <!-- Select All -->\r\n <div class=\"form-check px-0 mb-2 d-flex align-items-center\">\r\n <input\r\n type=\"checkbox\"\r\n class=\"form-check-input\"\r\n id=\"approver-all\"\r\n [checked]=\"areAllApproversSelected()\"\r\n (change)=\"onApproverSelectAll($event)\"\r\n />\r\n <label for=\"approver-all\" class=\"form-check-label ms-2\">\r\n Select All\r\n </label>\r\n </div>\r\n\r\n <!-- Options -->\r\n <div\r\n class=\"form-check mb-1 px-0 d-flex align-items-center\"\r\n *ngFor=\"let u of userList\"\r\n >\r\n <input\r\n class=\"form-check-input\"\r\n type=\"checkbox\"\r\n [id]=\"'app-' + u.email\"\r\n [checked]=\"isApproverSelected(u.email)\"\r\n (change)=\"onApproverChange($event, u.email)\"\r\n />\r\n <label class=\"form-check-label ms-2\" [for]=\"'app-' + u.email\">\r\n {{ u.email }}\r\n </label>\r\n </div>\r\n </div>\r\n</div>\r\n\r\n\r\n <!-- Footer buttons -->\r\n <div class=\"d-flex justify-content-between w-100 mt-4\">\r\n <button type=\"button\" class=\"btn btn-outline w-50 me-1\" (click)=\"onReset()\">\r\n Reset\r\n </button>\r\n <button type=\"button\" class=\"btn btn-primary w-50 ms-1\" (click)=\"onApply()\">\r\n Apply\r\n </button>\r\n </div>\r\n </form>\r\n </div>\r\n</div>\r\n", styles: [".filter-wrapper{position:absolute;top:0;right:0;padding:16px;z-index:1050}.filter-card{width:350px;background:#fff;border-radius:12px;padding:16px 18px;box-shadow:0 8px 30px #00000014;font-size:13px}.filter-header .title{font-weight:600;font-size:14px}.btn-close{border:none;background:transparent;font-size:16px;cursor:pointer}.form-label{font-size:12px;margin-bottom:4px}.form-select,.form-control{font-size:13px}.gap-2{gap:4px}.badge{padding:0 6px;border-radius:999px;font-size:11px;background:#e5f3ff;color:#007bff}input[type=checkbox]{width:16px!important;height:16px!important;margin:-2px 5px;border-radius:4px!important;-webkit-appearance:none;-moz-appearance:none;-o-appearance:none;appearance:none;outline:1px solid var(--gray-600, #D0D5DD)!important;box-shadow:none;font-size:.5em;text-align:center;line-height:1em;background:#fff}input[type=checkbox]{outline:1px solid var(--primary-600, #00A3FF)!important;background-color:var(--primary-50, #EAF8FF)}input[type=checkbox]:checked:after{content:\"\";transform:rotate(45deg);border-bottom:2px solid #00A3FF;border-right:2px solid #00A3FF;display:inline-block;width:.5em;padding-left:3px;padding-top:10px;padding-right:0}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i1.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.MinValidator, selector: "input[type=number][min][formControlName],input[type=number][min][formControl],input[type=number][min][ngModel]", inputs: ["min"] }, { kind: "directive", type: i1.MaxValidator, selector: "input[type=number][max][formControlName],input[type=number][max][formControl],input[type=number][max][ngModel]", inputs: ["max"] }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }] });
|
|
108
319
|
}
|
|
109
320
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TicketFilterPanelComponent, decorators: [{
|
|
110
321
|
type: Component,
|
|
111
|
-
args: [{ selector: 'lib-ticket-filter-panel', template: "<!-- ticket-filter-panel.component.html -->\r\n<div class=\"filter-wrapper\" *ngIf=\"isOpen\">\r\n <div class=\"filter-card\">\r\n <div class=\"filter-header d-flex justify-content-between align-items-center mb-3\">\r\n <div class=\"title\">Filter Options</div>\r\n <!-- <button type=\"button\" class=\"btn-close\" (click)=\"close()\">\u2715</button> -->\r\n </div>\r\n\r\n <form [formGroup]=\"filterForm\">\r\n <!-- Status -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Status</label>\r\n <select formControlName=\"status\" class=\"form-select\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let s of statusOptions\" [ngValue]=\"s\">{{ s }}</option>\r\n </select>\r\n </div>\r\n\r\n <!-- Reviewer accuracy -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Reviewer accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-50\" formControlName=\"reviewerCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <!-- one or two inputs -->\r\n <ng-container *ngIf=\"isBetween('reviewerCondition'); else singleReviewer\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleReviewer>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Approver accuracy -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Approver accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-50\" formControlName=\"approverCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <ng-container *ngIf=\"isBetween('approverCondition'); else singleApprover\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleApprover>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Tango accuracy -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Tango accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-50\" formControlName=\"tangoCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <ng-container *ngIf=\"isBetween('tangoCondition'); else singleTango\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleTango>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Approved By -->\r\n <div class=\"mb-3\">\r\n <label class=\"form-label\">Approved by</label>\r\n <input\r\n class=\"form-control\"\r\n formControlName=\"approvedBy\"\r\n placeholder=\"Select / search by email\" />\r\n </div>\r\n\r\n <!-- Footer buttons -->\r\n <div class=\"d-flex justify-content-between w-100 mt-4\">\r\n <button type=\"button\" class=\"btn btn-outline w-50 me-1\" (click)=\"onReset()\">\r\n Reset\r\n </button>\r\n <button type=\"button\" class=\"btn btn-primary w-50 ms-1\" (click)=\"onApply()\">\r\n Apply\r\n </button>\r\n </div>\r\n </form>\r\n </div>\r\n</div>\r\n", styles: [".filter-wrapper{position:absolute;top:0;right:0;padding:16px;z-index:1050}.filter-card{width:350px;background:#fff;border-radius:12px;padding:16px 18px;box-shadow:0 8px 30px #00000014;font-size:13px}.filter-header .title{font-weight:600;font-size:14px}.btn-close{border:none;background:transparent;font-size:16px;cursor:pointer}.form-label{font-size:12px;margin-bottom:4px}.form-select,.form-control{font-size:13px}.gap-2{gap:4px}\n"] }]
|
|
112
|
-
}], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i0.ElementRef }], propDecorators: { apply: [{
|
|
322
|
+
args: [{ selector: 'lib-ticket-filter-panel', template: "<!-- ticket-filter-panel.component.html -->\r\n<div class=\"filter-wrapper\" *ngIf=\"isOpen\">\r\n <div class=\"filter-card\">\r\n <div class=\"filter-header d-flex justify-content-between align-items-center mb-3\">\r\n <div class=\"title\">Filter Options</div>\r\n <!-- <button type=\"button\" class=\"btn-close\" (click)=\"close()\">\u2715</button> -->\r\n </div>\r\n\r\n <form [formGroup]=\"filterForm\">\r\n <!-- Status -->\r\n<div class=\"mb-3 position-relative\" (click)=\"$event.stopPropagation()\">\r\n <label class=\"form-label\">Status</label>\r\n\r\n <!-- Header / trigger -->\r\n <div\r\n class=\"status-select border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n (click)=\"toggleStatusDropdown($event)\"\r\n >\r\n <span *ngIf=\"!selectedStatuses.length\">Select</span>\r\n\r\n <span *ngIf=\"selectedStatuses.length === 1\">\r\n {{ selectedStatuses[0] }}\r\n </span>\r\n\r\n <span *ngIf=\"selectedStatuses.length > 1\" class=\"d-flex align-items-center\">\r\n <span>{{ selectedStatuses[0] }}</span>\r\n <span class=\"badge ms-2\">\r\n +{{ selectedStatuses.length - 1 }}\r\n </span>\r\n </span>\r\n\r\n </div>\r\n\r\n <!-- DROPDOWN PANEL (OVERLAY) -->\r\n <div\r\n class=\"status-dropdown position-absolute w-100 mt-1 border rounded p-1 bg-white\"\r\n style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n *ngIf=\"statusDropdownOpen\"\r\n (click)=\"$event.stopPropagation()\"\r\n >\r\n <!-- Select All -->\r\n <div class=\"form-check px-0 d-flex align-items-center mb-2\">\r\n <input\r\n type=\"checkbox\"\r\n class=\"form-check-input text-dark\"\r\n id=\"status-select-all\"\r\n [checked]=\"areAllStatusesSelected()\"\r\n (change)=\"onStatusSelectAllChange($event)\"\r\n />\r\n <label class=\"form-check-label ms-2\" for=\"status-select-all\">\r\n Select All\r\n </label>\r\n </div>\r\n\r\n\r\n <!-- Options -->\r\n <div\r\n class=\"form-check px-0 d-flex align-items-center mb-1\"\r\n *ngFor=\"let s of statusOptions\"\r\n >\r\n <input\r\n type=\"checkbox\"\r\n class=\"form-check-input text-dark\"\r\n [id]=\"'status-' + s\"\r\n [checked]=\"isStatusSelected(s)\"\r\n (change)=\"onStatusChange($event, s)\"\r\n />\r\n <label class=\"form-check-label ms-2\" [for]=\"'status-' + s\">\r\n {{ s }}\r\n </label>\r\n </div>\r\n </div>\r\n</div>\r\n\r\n\r\n\r\n <!-- Reviewer accuracy -->\r\n <div class=\"mb-3\" *ngIf=\"showReviewerAccuracy\">\r\n <label class=\"form-label\">Reviewer accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-100\" formControlName=\"reviewerCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <!-- one or two inputs -->\r\n <ng-container *ngIf=\"isBetween('reviewerCondition'); else singleReviewer\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleReviewer>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"reviewerFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Approver accuracy -->\r\n <div class=\"mb-3\" *ngIf=\"showApproverAccuracy\">\r\n <label class=\"form-label\">Approver accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-100\" formControlName=\"approverCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <ng-container *ngIf=\"isBetween('approverCondition'); else singleApprover\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleApprover>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"approverFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n\r\n <!-- Tango accuracy -->\r\n <div class=\"mb-3\" *ngIf=\"showTangoAccuracy\">\r\n <label class=\"form-label\">Tango accuracy (%) by condition</label>\r\n <div class=\"d-flex gap-2\">\r\n <select class=\"form-select w-50\" formControlName=\"tangoCondition\">\r\n <option [ngValue]=\"null\">Select</option>\r\n <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n {{ c.label }}\r\n </option>\r\n </select>\r\n\r\n <ng-container *ngIf=\"isBetween('tangoCondition'); else singleTango\">\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoFrom\"\r\n placeholder=\"From\" />\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoTo\"\r\n placeholder=\"To\" />\r\n </ng-container>\r\n <ng-template #singleTango>\r\n <input\r\n type=\"number\"\r\n min=\"1\"\r\n max=\"100\"\r\n class=\"form-control\"\r\n formControlName=\"tangoFrom\"\r\n placeholder=\"1 to 100%\" />\r\n </ng-template>\r\n </div>\r\n </div>\r\n<div\r\n class=\"mb-3 position-relative\"\r\n *ngIf=\"showReviewedBy\"\r\n (click)=\"$event.stopPropagation()\"\r\n>\r\n <label class=\"form-label\">Reviewed by</label>\r\n\r\n <!-- Trigger / display -->\r\n <div\r\n class=\"border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n (click)=\"toggleReviewerDropdown($event)\"\r\n >\r\n <!-- 0 selected -->\r\n <span *ngIf=\"!selectedReviewedBy.length\">\r\n Select\r\n </span>\r\n\r\n <!-- 1 selected -->\r\n <span *ngIf=\"selectedReviewedBy.length === 1\">\r\n {{ selectedReviewedBy[0] }}\r\n </span>\r\n\r\n <!-- >1 selected -->\r\n <span\r\n *ngIf=\"selectedReviewedBy.length > 1\"\r\n class=\"d-flex align-items-center\"\r\n >\r\n <span>{{ selectedReviewedBy[0] }}</span>\r\n <span class=\"badge ms-2\">\r\n +{{ selectedReviewedBy.length - 1 }}\r\n </span>\r\n </span>\r\n </div>\r\n\r\n <!-- Dropdown panel -->\r\n <div\r\n *ngIf=\"reviewerDropdownOpen\"\r\n class=\"position-absolute w-100 border rounded bg-white p-1 mt-1\"\r\n style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n (click)=\"$event.stopPropagation()\"\r\n >\r\n <!-- Select All -->\r\n <div class=\"form-check px-0 mb-2 d-flex align-items-center\">\r\n <input\r\n type=\"checkbox\"\r\n class=\"form-check-input\"\r\n id=\"reviewer-all\"\r\n [checked]=\"areAllReviewersSelected()\"\r\n (change)=\"onReviewerSelectAll($event)\"\r\n />\r\n <label for=\"reviewer-all\" class=\"form-check-label ms-2\">\r\n Select All\r\n </label>\r\n </div>\r\n\r\n <!-- Options -->\r\n <div\r\n class=\"form-check mb-1 px-0 d-flex align-items-center\"\r\n *ngFor=\"let u of userList\"\r\n >\r\n <input\r\n class=\"form-check-input\"\r\n type=\"checkbox\"\r\n [id]=\"'rev-' + u.email\"\r\n [checked]=\"isReviewerSelected(u.email)\"\r\n (change)=\"onReviewerChange($event, u.email)\"\r\n />\r\n <label class=\"form-check-label ms-2\" [for]=\"'rev-' + u.email\">\r\n {{ u.email }}\r\n </label>\r\n </div>\r\n </div>\r\n</div>\r\n\r\n\r\n <!-- Approved By -->\r\n<div\r\n class=\"mb-3 position-relative\"\r\n *ngIf=\"showApprovedBy\"\r\n (click)=\"$event.stopPropagation()\"\r\n>\r\n <label class=\"form-label\">Approved by</label>\r\n\r\n <!-- Trigger / display -->\r\n <div\r\n class=\"border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n (click)=\"toggleApproverDropdown($event)\"\r\n >\r\n <!-- 0 selected -->\r\n <span *ngIf=\"!selectedApprovedBy.length\">\r\n Select\r\n </span>\r\n\r\n <!-- 1 selected -->\r\n <span *ngIf=\"selectedApprovedBy.length === 1\">\r\n {{ selectedApprovedBy[0] }}\r\n </span>\r\n\r\n <!-- >1 selected -->\r\n <span\r\n *ngIf=\"selectedApprovedBy.length > 1\"\r\n class=\"d-flex align-items-center\"\r\n >\r\n <span>{{ selectedApprovedBy[0] }}</span>\r\n <span class=\"badge ms-2\">\r\n +{{ selectedApprovedBy.length - 1 }}\r\n </span>\r\n </span>\r\n\r\n </div>\r\n\r\n <!-- Dropdown panel -->\r\n <div\r\n *ngIf=\"approverDropdownOpen\"\r\n class=\"position-absolute w-100 border rounded bg-white p-2 mt-1\"\r\n style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n (click)=\"$event.stopPropagation()\"\r\n >\r\n <!-- Select All -->\r\n <div class=\"form-check px-0 mb-2 d-flex align-items-center\">\r\n <input\r\n type=\"checkbox\"\r\n class=\"form-check-input\"\r\n id=\"approver-all\"\r\n [checked]=\"areAllApproversSelected()\"\r\n (change)=\"onApproverSelectAll($event)\"\r\n />\r\n <label for=\"approver-all\" class=\"form-check-label ms-2\">\r\n Select All\r\n </label>\r\n </div>\r\n\r\n <!-- Options -->\r\n <div\r\n class=\"form-check mb-1 px-0 d-flex align-items-center\"\r\n *ngFor=\"let u of userList\"\r\n >\r\n <input\r\n class=\"form-check-input\"\r\n type=\"checkbox\"\r\n [id]=\"'app-' + u.email\"\r\n [checked]=\"isApproverSelected(u.email)\"\r\n (change)=\"onApproverChange($event, u.email)\"\r\n />\r\n <label class=\"form-check-label ms-2\" [for]=\"'app-' + u.email\">\r\n {{ u.email }}\r\n </label>\r\n </div>\r\n </div>\r\n</div>\r\n\r\n\r\n <!-- Footer buttons -->\r\n <div class=\"d-flex justify-content-between w-100 mt-4\">\r\n <button type=\"button\" class=\"btn btn-outline w-50 me-1\" (click)=\"onReset()\">\r\n Reset\r\n </button>\r\n <button type=\"button\" class=\"btn btn-primary w-50 ms-1\" (click)=\"onApply()\">\r\n Apply\r\n </button>\r\n </div>\r\n </form>\r\n </div>\r\n</div>\r\n", styles: [".filter-wrapper{position:absolute;top:0;right:0;padding:16px;z-index:1050}.filter-card{width:350px;background:#fff;border-radius:12px;padding:16px 18px;box-shadow:0 8px 30px #00000014;font-size:13px}.filter-header .title{font-weight:600;font-size:14px}.btn-close{border:none;background:transparent;font-size:16px;cursor:pointer}.form-label{font-size:12px;margin-bottom:4px}.form-select,.form-control{font-size:13px}.gap-2{gap:4px}.badge{padding:0 6px;border-radius:999px;font-size:11px;background:#e5f3ff;color:#007bff}input[type=checkbox]{width:16px!important;height:16px!important;margin:-2px 5px;border-radius:4px!important;-webkit-appearance:none;-moz-appearance:none;-o-appearance:none;appearance:none;outline:1px solid var(--gray-600, #D0D5DD)!important;box-shadow:none;font-size:.5em;text-align:center;line-height:1em;background:#fff}input[type=checkbox]{outline:1px solid var(--primary-600, #00A3FF)!important;background-color:var(--primary-50, #EAF8FF)}input[type=checkbox]:checked:after{content:\"\";transform:rotate(45deg);border-bottom:2px solid #00A3FF;border-right:2px solid #00A3FF;display:inline-block;width:.5em;padding-left:3px;padding-top:10px;padding-right:0}\n"] }]
|
|
323
|
+
}], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i0.ElementRef }, { type: i2.TicketService }], propDecorators: { apply: [{
|
|
113
324
|
type: Output
|
|
114
325
|
}], panelClosed: [{
|
|
115
326
|
type: Output
|
|
327
|
+
}], permissionType: [{
|
|
328
|
+
type: Input
|
|
329
|
+
}], userType: [{
|
|
330
|
+
type: Input
|
|
331
|
+
}], client: [{
|
|
332
|
+
type: Input
|
|
116
333
|
}], onDocumentClick: [{
|
|
117
334
|
type: HostListener,
|
|
118
335
|
args: ['document:click', ['$event']]
|
|
119
336
|
}] } });
|
|
120
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ticket-filter-panel.component.js","sourceRoot":"","sources":["../../../../../../projects/tango-manage-tickets/src/lib/components/ticket-filter-panel/ticket-filter-panel.component.ts","../../../../../../projects/tango-manage-tickets/src/lib/components/ticket-filter-panel/ticket-filter-panel.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, Output } from '@angular/core';\r\nimport { FormBuilder, FormGroup } from '@angular/forms';\r\n\r\ntype Condition = 'gt' | 'lt' | 'gte' | 'lte' | 'between';\r\n\r\n@Component({\r\n  selector: 'lib-ticket-filter-panel',\r\n  templateUrl: './ticket-filter-panel.component.html',\r\n  styleUrl: './ticket-filter-panel.component.scss'\r\n})\r\nexport class TicketFilterPanelComponent {\r\n   @Output() apply = new EventEmitter<any>();\r\n  @Output() panelClosed = new EventEmitter<void>();   // ✅ not `close`\r\n  @HostListener('document:click', ['$event'])\r\n  onDocumentClick(event: MouseEvent): void {\r\n    if (!this.isOpen) return;\r\n\r\n    const clickedInside = this.eRef.nativeElement.contains(event.target);\r\n    if (!clickedInside) {\r\n      this.close();\r\n    }\r\n  }\r\n  isOpen = false;                                     // controls visibility\r\n\r\n  // called from parent via #filterPanel.open()\r\n  open() {\r\n    this.isOpen = true;\r\n  }\r\n\r\n  // called from close button inside the panel\r\n  close() {\r\n    this.isOpen = false;\r\n    this.panelClosed.emit();\r\n  }\r\n\r\n  showPanel = false;           // you can also control this from parent\r\n\r\n  statusOptions = [\r\n    'Open',\r\n    'In-Progress',\r\n    'Close',\r\n    'Under tango review',\r\n    'Tango review done',\r\n    'Expired',\r\n  ];\r\n\r\n  conditionOptions = [\r\n    { value: 'gt', label: '> Greater than' },\r\n    { value: 'lt', label: '< Lesser than' },\r\n    { value: 'gte', label: '>= Greater than or equal to' },\r\n    { value: 'lte', label: '<= Lesser than or equal to' },\r\n    { value: 'between', label: 'Between' },\r\n  ];\r\n\r\n  filterForm: FormGroup;\r\n\r\n  constructor(private fb: FormBuilder,private eRef: ElementRef) {\r\n    this.filterForm = this.fb.group({\r\n      status: ['Open'],\r\n\r\n      reviewerCondition: ['gte' as Condition],\r\n      reviewerFrom: [85],\r\n      reviewerTo: [null],\r\n\r\n      approverCondition: [null as Condition | null],\r\n      approverFrom: [null],\r\n      approverTo: [null],\r\n\r\n      tangoCondition: [null as Condition | null],\r\n      tangoFrom: [null],\r\n      tangoTo: [null],\r\n\r\n      approvedBy: [null],\r\n    });\r\n  }\r\n\r\n\r\n\r\n  isBetween(controlName: string): boolean {\r\n    return this.filterForm.get(controlName)?.value === 'between';\r\n  }\r\n\r\n  onReset(): void {\r\n    this.filterForm.reset({\r\n      status: null,\r\n      reviewerCondition: null,\r\n      reviewerFrom: null,\r\n      reviewerTo: null,\r\n      approverCondition: null,\r\n      approverFrom: null,\r\n      approverTo: null,\r\n      tangoCondition: null,\r\n      tangoFrom: null,\r\n      tangoTo: null,\r\n      approvedBy: null,\r\n    });\r\n      this.close();\r\n  }\r\n\r\n  onApply(): void {\r\n    const raw = this.filterForm.value;\r\n\r\n    // build a clean DTO for API / table filter\r\n    const payload = {\r\n      filterByStatus: raw.status || null,\r\n      filterByReviewer: this.mapAccuracy(raw.reviewerCondition, raw.reviewerFrom, raw.reviewerTo),\r\n      filterByApprover: this.mapAccuracy(raw.approverCondition, raw.approverFrom, raw.approverTo),\r\n      filterByTango: this.mapAccuracy(raw.tangoCondition, raw.tangoFrom, raw.tangoTo),\r\n      filterByReviewedBy: raw.reviewedBy || null,\r\n      fileterByApprovedBy: raw.approvedBy || null,\r\n    };\r\n\r\n    this.apply.emit(payload);\r\n     this.close();\r\n    this.showPanel = false;\r\n  }\r\n\r\n  private mapAccuracy(\r\n    condition: Condition | null,\r\n    from: number | null,\r\n    to: number | null\r\n  ) {\r\n    if (!condition || from == null) return null;\r\n\r\n    if (condition === 'between') {\r\n      if (to == null) return null;\r\n      return { condition, from, to };\r\n    }\r\n\r\n    return { condition, value: from };\r\n  }\r\n}\r\n","<!-- ticket-filter-panel.component.html -->\r\n<div class=\"filter-wrapper\" *ngIf=\"isOpen\">\r\n  <div class=\"filter-card\">\r\n    <div class=\"filter-header d-flex justify-content-between align-items-center mb-3\">\r\n  <div class=\"title\">Filter Options</div>\r\n      <!-- <button type=\"button\" class=\"btn-close\" (click)=\"close()\">✕</button> -->\r\n    </div>\r\n\r\n    <form [formGroup]=\"filterForm\">\r\n      <!-- Status -->\r\n      <div class=\"mb-3\">\r\n        <label class=\"form-label\">Status</label>\r\n        <select formControlName=\"status\" class=\"form-select\">\r\n          <option [ngValue]=\"null\">Select</option>\r\n          <option *ngFor=\"let s of statusOptions\" [ngValue]=\"s\">{{ s }}</option>\r\n        </select>\r\n      </div>\r\n\r\n      <!-- Reviewer accuracy -->\r\n      <div class=\"mb-3\">\r\n        <label class=\"form-label\">Reviewer accuracy (%) by condition</label>\r\n        <div class=\"d-flex gap-2\">\r\n          <select class=\"form-select w-50\" formControlName=\"reviewerCondition\">\r\n            <option [ngValue]=\"null\">Select</option>\r\n            <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n              {{ c.label }}\r\n            </option>\r\n          </select>\r\n\r\n          <!-- one or two inputs -->\r\n          <ng-container *ngIf=\"isBetween('reviewerCondition'); else singleReviewer\">\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"reviewerFrom\"\r\n              placeholder=\"From\" />\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"reviewerTo\"\r\n              placeholder=\"To\" />\r\n          </ng-container>\r\n          <ng-template #singleReviewer>\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"reviewerFrom\"\r\n              placeholder=\"1 to 100%\" />\r\n          </ng-template>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Approver accuracy -->\r\n      <div class=\"mb-3\">\r\n        <label class=\"form-label\">Approver accuracy (%) by condition</label>\r\n        <div class=\"d-flex gap-2\">\r\n          <select class=\"form-select w-50\" formControlName=\"approverCondition\">\r\n            <option [ngValue]=\"null\">Select</option>\r\n            <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n              {{ c.label }}\r\n            </option>\r\n          </select>\r\n\r\n          <ng-container *ngIf=\"isBetween('approverCondition'); else singleApprover\">\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"approverFrom\"\r\n              placeholder=\"From\" />\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"approverTo\"\r\n              placeholder=\"To\" />\r\n          </ng-container>\r\n          <ng-template #singleApprover>\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"approverFrom\"\r\n              placeholder=\"1 to 100%\" />\r\n          </ng-template>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Tango accuracy -->\r\n      <div class=\"mb-3\">\r\n        <label class=\"form-label\">Tango accuracy (%) by condition</label>\r\n        <div class=\"d-flex gap-2\">\r\n          <select class=\"form-select w-50\" formControlName=\"tangoCondition\">\r\n            <option [ngValue]=\"null\">Select</option>\r\n            <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n              {{ c.label }}\r\n            </option>\r\n          </select>\r\n\r\n          <ng-container *ngIf=\"isBetween('tangoCondition'); else singleTango\">\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"tangoFrom\"\r\n              placeholder=\"From\" />\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"tangoTo\"\r\n              placeholder=\"To\" />\r\n          </ng-container>\r\n          <ng-template #singleTango>\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"tangoFrom\"\r\n              placeholder=\"1 to 100%\" />\r\n          </ng-template>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Approved By -->\r\n      <div class=\"mb-3\">\r\n        <label class=\"form-label\">Approved by</label>\r\n        <input\r\n          class=\"form-control\"\r\n          formControlName=\"approvedBy\"\r\n          placeholder=\"Select / search by email\" />\r\n      </div>\r\n\r\n      <!-- Footer buttons -->\r\n      <div class=\"d-flex justify-content-between w-100 mt-4\">\r\n        <button type=\"button\" class=\"btn btn-outline w-50 me-1\" (click)=\"onReset()\">\r\n          Reset\r\n        </button>\r\n        <button type=\"button\" class=\"btn btn-primary w-50 ms-1\" (click)=\"onApply()\">\r\n          Apply\r\n        </button>\r\n      </div>\r\n    </form>\r\n  </div>\r\n</div>\r\n"]}
|
|
337
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ticket-filter-panel.component.js","sourceRoot":"","sources":["../../../../../../projects/tango-manage-tickets/src/lib/components/ticket-filter-panel/ticket-filter-panel.component.ts","../../../../../../projects/tango-manage-tickets/src/lib/components/ticket-filter-panel/ticket-filter-panel.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, OnChanges, Output, SimpleChanges } from '@angular/core';\r\nimport { FormBuilder, FormGroup } from '@angular/forms';\r\nimport { TicketService } from '../../services/ticket.service';\r\n\r\ntype Condition = 'gt' | 'lt' | 'gte' | 'lte' | 'between';\r\n\r\n@Component({\r\n  selector: 'lib-ticket-filter-panel',\r\n  templateUrl: './ticket-filter-panel.component.html',\r\n  styleUrl: './ticket-filter-panel.component.scss'\r\n})\r\nexport class TicketFilterPanelComponent implements OnChanges {\r\n   @Output() apply = new EventEmitter<any>();\r\n  @Output() panelClosed = new EventEmitter<void>();   // ✅ not `close`\r\n  @Input() permissionType: string | null = null;\r\n@Input() userType: string | null = null; // e.g. 'tango' or others\r\n@Input() client :any;\r\n  @HostListener('document:click', ['$event'])\r\n  onDocumentClick(event: MouseEvent): void {\r\n    if (!this.isOpen) return;\r\n\r\n    const clickedInside = this.eRef.nativeElement.contains(event.target);\r\n    if (!clickedInside) {\r\n      this.close();\r\n    }\r\n  }\r\n  isOpen = false;                                     // controls visibility\r\n\r\n  // called from parent via #filterPanel.open()\r\n  open() {\r\n    this.isOpen = true;\r\n  }\r\n\r\n  // called from close button inside the panel\r\n  close() {\r\n    this.isOpen = false;\r\n    this.panelClosed.emit();\r\n  }\r\n\r\n  showPanel = false;           // you can also control this from parent\r\n\r\nstatusOptions: any[] = [\r\n  'Open',\r\n  'In-Progress',\r\n  'Close',\r\n  'Under tango review',\r\n  'Tango review done',\r\n  'Expired',\r\n  'Tango Expired',\r\n];\r\n\r\n\r\n  conditionOptions = [\r\n    { value: 'gt', label: '> Greater than' },\r\n    { value: 'lt', label: '< Lesser than' },\r\n    { value: 'gte', label: '>= Greater than or equal to' },\r\n    { value: 'lte', label: '<= Lesser than or equal to' },\r\n    { value: 'between', label: 'Between' },\r\n  ];\r\n\r\n  filterForm: FormGroup;\r\n\r\n  constructor(private fb: FormBuilder,private eRef: ElementRef,private service:TicketService) {\r\n    this.filterForm = this.fb.group({\r\n      status: [[]],\r\n\r\n      reviewerCondition: [null as Condition | null],\r\n      reviewerFrom: [null],\r\n      reviewerTo: [null],\r\n\r\n      approverCondition: [null as Condition | null],\r\n      approverFrom: [null],\r\n      approverTo: [null],\r\n\r\n      tangoCondition: [null as Condition | null],\r\n      tangoFrom: [null],\r\n      tangoTo: [null],\r\n\r\n      reviewedBy: [[]],\r\n  approvedBy: [[]],  \r\n    });\r\n  }\r\n\r\nget isTango(): boolean {\r\n  return this.userType === 'tango';\r\n}\r\n\r\n// SHOW / HIDE RULES\r\n\r\n// Reviewer accuracy and \"Reviewed by\" visible for tango + reviewer\r\nget showReviewerAccuracy(): boolean {\r\n  return this.isTango || this.permissionType === 'review';\r\n}\r\n\r\nget showReviewedBy(): boolean {\r\n  return this.isTango || this.permissionType === 'review';\r\n}\r\n\r\n// Approver accuracy and \"Approved by\" visible for tango + approver\r\nget showApproverAccuracy(): boolean {\r\n  return this.isTango || this.permissionType === 'approve';\r\n}\r\n\r\nget showApprovedBy(): boolean {\r\n  return this.isTango || this.permissionType === 'approve';\r\n}\r\n\r\n// Tango accuracy only for tango users\r\nget showTangoAccuracy(): boolean {\r\n  return this.isTango;\r\n}\r\nstatusDropdownOpen = false;\r\nisStatusSelected(status: string): boolean {\r\n  const selected: string[] = this.filterForm.value.status || [];\r\n  return selected.includes(status);\r\n}\r\n\r\nonStatusChange(event: Event, status: string) {\r\n  const checkbox = event.target as HTMLInputElement;\r\n  const selected: string[] = this.filterForm.value.status || [];\r\n  let next: string[];\r\n\r\n  if (checkbox.checked) {\r\n    next = selected.includes(status) ? selected : [...selected, status];\r\n  } else {\r\n    next = selected.filter(s => s !== status);\r\n  }\r\n\r\n  this.filterForm.patchValue({ status: next });\r\n}\r\n\r\ntoggleStatusDropdown(event?: MouseEvent) {\r\n  if (event) {\r\n    event.stopPropagation(); // avoid document click closing it\r\n  }\r\n  this.statusDropdownOpen = !this.statusDropdownOpen;\r\n}\r\n\r\n// Are all statuses currently selected?\r\nareAllStatusesSelected(): boolean {\r\n  const selected: string[] = this.filterForm.value.status || [];\r\n  return selected.length > 0 && selected.length === this.statusOptions.length;\r\n}\r\n\r\n// Handle \"Select All\" checkbox change\r\nonStatusSelectAllChange(event: Event) {\r\n  const checkbox = event.target as HTMLInputElement;\r\n\r\n  if (checkbox.checked) {\r\n    // Select every status\r\n    this.filterForm.patchValue({ status: [...this.statusOptions] });\r\n  } else {\r\n    // Clear all\r\n    this.filterForm.patchValue({ status: [] });\r\n  }\r\n}\r\nget selectedStatuses(): string[] {\r\n  return this.filterForm.value.status || [];\r\n}\r\n\r\n  isBetween(controlName: string): boolean {\r\n    return this.filterForm.get(controlName)?.value === 'between';\r\n  }\r\nngOnChanges(changes: SimpleChanges): void {\r\n  if (changes['client'] || changes['permissionType'] || changes['userType']) {\r\n    this.loadReviewerList();\r\n  }\r\n}\r\nuserList:any\r\nprivate loadReviewerList(): void {\r\n  if (!this.client) {\r\n    this.userList = [];\r\n    return;\r\n  }\r\n\r\n  let type: string;\r\n\r\n  if (this.userType === 'tango') {\r\n    type = 'tango';    // or 'all'\r\n  } else if (this.permissionType === 'review') {\r\n    type = 'review';\r\n  } else if (this.permissionType === 'approve') {\r\n    type = 'approve';\r\n  } else {\r\n    this.userList = [];\r\n    return;\r\n  }\r\n\r\n  this.service.getReviewerApi(this.client, type).subscribe({\r\n    next: (res: any) => {\r\n      if (res && res.code === 200) {\r\n        this.userList = res.data;\r\n      } else {\r\n        this.userList = [];\r\n      }\r\n    },\r\n    error: () => {\r\n      this.userList = [];\r\n    }\r\n  });\r\n}\r\n\r\n// open flags\r\nreviewerDropdownOpen = false;\r\napproverDropdownOpen = false;\r\n\r\n// convenience getters\r\nget selectedReviewedBy(): string[] {\r\n  return this.filterForm.value.reviewedBy || [];\r\n}\r\n\r\nget selectedApprovedBy(): string[] {\r\n  return this.filterForm.value.approvedBy || [];\r\n}\r\n\r\n// --- Reviewed By helpers ---\r\n\r\ntoggleReviewerDropdown(event?: MouseEvent) {\r\n  if (event) event.stopPropagation();\r\n  this.reviewerDropdownOpen = !this.reviewerDropdownOpen;\r\n}\r\n\r\nisReviewerSelected(email: string): boolean {\r\n  return this.selectedReviewedBy.includes(email);\r\n}\r\n\r\nonReviewerChange(event: Event, email: string) {\r\n  const checkbox = event.target as HTMLInputElement;\r\n  const selected = [...this.selectedReviewedBy];\r\n\r\n  if (checkbox.checked) {\r\n    if (!selected.includes(email)) selected.push(email);\r\n  } else {\r\n    const i = selected.indexOf(email);\r\n    if (i >= 0) selected.splice(i, 1);\r\n  }\r\n\r\n  this.filterForm.patchValue({ reviewedBy: selected });\r\n}\r\n\r\nareAllReviewersSelected(): boolean {\r\n  return (\r\n    this.userList.length > 0 &&\r\n    this.selectedReviewedBy.length === this.userList.length\r\n  );\r\n}\r\n\r\nonReviewerSelectAll(event: Event) {\r\n  const checkbox = event.target as HTMLInputElement;\r\n\r\n  if (checkbox.checked) {\r\n    this.filterForm.patchValue({\r\n      reviewedBy: this.userList.map((u:any) => u.email),\r\n    });\r\n  } else {\r\n    this.filterForm.patchValue({ reviewedBy: [] });\r\n  }\r\n}\r\n\r\n// --- Approved By helpers ---\r\n\r\ntoggleApproverDropdown(event?: MouseEvent) {\r\n  if (event) event.stopPropagation();\r\n  this.approverDropdownOpen = !this.approverDropdownOpen;\r\n}\r\n\r\nisApproverSelected(email: string): boolean {\r\n  return this.selectedApprovedBy.includes(email);\r\n}\r\n\r\nonApproverChange(event: Event, email: string) {\r\n  const checkbox = event.target as HTMLInputElement;\r\n  const selected = [...this.selectedApprovedBy];\r\n\r\n  if (checkbox.checked) {\r\n    if (!selected.includes(email)) selected.push(email);\r\n  } else {\r\n    const i = selected.indexOf(email);\r\n    if (i >= 0) selected.splice(i, 1);\r\n  }\r\n\r\n  this.filterForm.patchValue({ approvedBy: selected });\r\n}\r\n\r\nareAllApproversSelected(): boolean {\r\n  return (\r\n    this.userList.length > 0 &&\r\n    this.selectedApprovedBy.length === this.userList.length\r\n  );\r\n}\r\n\r\nonApproverSelectAll(event: Event) {\r\n  const checkbox = event.target as HTMLInputElement;\r\n\r\n  if (checkbox.checked) {\r\n    this.filterForm.patchValue({\r\n      approvedBy: this.userList.map((u:any) => u.email),\r\n    });\r\n  } else {\r\n    this.filterForm.patchValue({ approvedBy: [] });\r\n  }\r\n}\r\n\r\n\r\n  onReset(): void {\r\n    this.filterForm.reset({\r\n      status: [],\r\n  reviewerCondition: null,\r\n  reviewerFrom: null,\r\n  reviewerTo: null,\r\n  approverCondition: null,\r\n  approverFrom: null,\r\n  approverTo: null,\r\n  tangoCondition: null,\r\n  tangoFrom: null,\r\n  tangoTo: null,\r\n  reviewedBy: [],\r\n  approvedBy: [],\r\n    });\r\n      this.close();\r\n  }\r\nprivate mapAccuracy(\r\n  condition: Condition | null,\r\n  from: number | null,\r\n  to: number | null\r\n): string | null {\r\n\r\n  // nothing selected\r\n  if (!condition || from == null) return null;\r\n\r\n  // \"between\" → \"90 to 100\"\r\n  if (condition === 'between') {\r\n    if (to == null) return null;\r\n    return `${from} to ${to}`;\r\n  }\r\n\r\n  // other conditions (gt, lt, gte, lte)\r\n  const opMap: Record<Condition, string> = {\r\n    gt: '>',\r\n    lt: '<',\r\n    gte: '>=',\r\n    lte: '<=',\r\n    between: '', // already handled above\r\n  };\r\n\r\n  return `${opMap[condition]} ${from}`;  // e.g. \">= 90\"\r\n}\r\n\r\n  onApply(): void {\r\n\r\n   const raw = this.filterForm.value;\r\n\r\n  const reviewerStr = this.mapAccuracy(raw.reviewerCondition, raw.reviewerFrom, raw.reviewerTo);\r\n  const approverStr = this.mapAccuracy(raw.approverCondition, raw.approverFrom, raw.approverTo);\r\n  const tangoStr    = this.mapAccuracy(raw.tangoCondition, raw.tangoFrom, raw.tangoTo);\r\n\r\n  const payload = {\r\n    filterByStatus: raw.status || [],\r\n\r\n    filterByReviewer: reviewerStr,\r\n    filterByApprover: approverStr,\r\n    filterByTango: tangoStr,\r\n\r\n    filterByReviewedBy: raw.reviewedBy || [],        \r\n    fileterByApprovedBy: raw.approvedBy || [],        \r\n  };\r\n\r\n  this.apply.emit(payload);\r\n  this.close();\r\n    this.showPanel = false;\r\n  }\r\n\r\n\r\n}\r\n","<!-- ticket-filter-panel.component.html -->\r\n<div class=\"filter-wrapper\" *ngIf=\"isOpen\">\r\n  <div class=\"filter-card\">\r\n    <div class=\"filter-header d-flex justify-content-between align-items-center mb-3\">\r\n  <div class=\"title\">Filter Options</div>\r\n      <!-- <button type=\"button\" class=\"btn-close\" (click)=\"close()\">✕</button> -->\r\n    </div>\r\n\r\n    <form [formGroup]=\"filterForm\">\r\n      <!-- Status -->\r\n<div class=\"mb-3 position-relative\" (click)=\"$event.stopPropagation()\">\r\n  <label class=\"form-label\">Status</label>\r\n\r\n  <!-- Header / trigger -->\r\n  <div\r\n    class=\"status-select border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n    (click)=\"toggleStatusDropdown($event)\"\r\n  >\r\n    <span *ngIf=\"!selectedStatuses.length\">Select</span>\r\n\r\n    <span *ngIf=\"selectedStatuses.length === 1\">\r\n      {{ selectedStatuses[0] }}\r\n    </span>\r\n\r\n    <span *ngIf=\"selectedStatuses.length > 1\" class=\"d-flex align-items-center\">\r\n      <span>{{ selectedStatuses[0] }}</span>\r\n      <span class=\"badge ms-2\">\r\n        +{{ selectedStatuses.length - 1 }}\r\n      </span>\r\n    </span>\r\n\r\n  </div>\r\n\r\n  <!-- DROPDOWN PANEL (OVERLAY) -->\r\n  <div\r\n    class=\"status-dropdown position-absolute w-100 mt-1 border rounded p-1 bg-white\"\r\n    style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n    *ngIf=\"statusDropdownOpen\"\r\n    (click)=\"$event.stopPropagation()\"\r\n  >\r\n    <!-- Select All -->\r\n    <div class=\"form-check px-0 d-flex align-items-center mb-2\">\r\n      <input\r\n        type=\"checkbox\"\r\n        class=\"form-check-input text-dark\"\r\n        id=\"status-select-all\"\r\n        [checked]=\"areAllStatusesSelected()\"\r\n        (change)=\"onStatusSelectAllChange($event)\"\r\n      />\r\n      <label class=\"form-check-label ms-2\" for=\"status-select-all\">\r\n        Select All\r\n      </label>\r\n    </div>\r\n\r\n\r\n    <!-- Options -->\r\n    <div\r\n      class=\"form-check px-0 d-flex align-items-center mb-1\"\r\n      *ngFor=\"let s of statusOptions\"\r\n    >\r\n      <input\r\n        type=\"checkbox\"\r\n        class=\"form-check-input text-dark\"\r\n        [id]=\"'status-' + s\"\r\n        [checked]=\"isStatusSelected(s)\"\r\n        (change)=\"onStatusChange($event, s)\"\r\n      />\r\n      <label class=\"form-check-label ms-2\" [for]=\"'status-' + s\">\r\n        {{ s }}\r\n      </label>\r\n    </div>\r\n  </div>\r\n</div>\r\n\r\n\r\n\r\n      <!-- Reviewer accuracy -->\r\n      <div class=\"mb-3\" *ngIf=\"showReviewerAccuracy\">\r\n        <label class=\"form-label\">Reviewer accuracy (%) by condition</label>\r\n        <div class=\"d-flex gap-2\">\r\n          <select class=\"form-select w-100\" formControlName=\"reviewerCondition\">\r\n            <option [ngValue]=\"null\">Select</option>\r\n            <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n              {{ c.label }}\r\n            </option>\r\n          </select>\r\n\r\n          <!-- one or two inputs -->\r\n          <ng-container *ngIf=\"isBetween('reviewerCondition'); else singleReviewer\">\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"reviewerFrom\"\r\n              placeholder=\"From\" />\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"reviewerTo\"\r\n              placeholder=\"To\" />\r\n          </ng-container>\r\n          <ng-template #singleReviewer>\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"reviewerFrom\"\r\n              placeholder=\"1 to 100%\" />\r\n          </ng-template>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Approver accuracy -->\r\n      <div class=\"mb-3\" *ngIf=\"showApproverAccuracy\">\r\n        <label class=\"form-label\">Approver accuracy (%) by condition</label>\r\n        <div class=\"d-flex gap-2\">\r\n          <select class=\"form-select w-100\" formControlName=\"approverCondition\">\r\n            <option [ngValue]=\"null\">Select</option>\r\n            <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n              {{ c.label }}\r\n            </option>\r\n          </select>\r\n\r\n          <ng-container *ngIf=\"isBetween('approverCondition'); else singleApprover\">\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"approverFrom\"\r\n              placeholder=\"From\" />\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"approverTo\"\r\n              placeholder=\"To\" />\r\n          </ng-container>\r\n          <ng-template #singleApprover>\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"approverFrom\"\r\n              placeholder=\"1 to 100%\" />\r\n          </ng-template>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Tango accuracy -->\r\n      <div class=\"mb-3\" *ngIf=\"showTangoAccuracy\">\r\n        <label class=\"form-label\">Tango accuracy (%) by condition</label>\r\n        <div class=\"d-flex gap-2\">\r\n          <select class=\"form-select w-50\" formControlName=\"tangoCondition\">\r\n            <option [ngValue]=\"null\">Select</option>\r\n            <option *ngFor=\"let c of conditionOptions\" [ngValue]=\"c.value\">\r\n              {{ c.label }}\r\n            </option>\r\n          </select>\r\n\r\n          <ng-container *ngIf=\"isBetween('tangoCondition'); else singleTango\">\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"tangoFrom\"\r\n              placeholder=\"From\" />\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"tangoTo\"\r\n              placeholder=\"To\" />\r\n          </ng-container>\r\n          <ng-template #singleTango>\r\n            <input\r\n              type=\"number\"\r\n              min=\"1\"\r\n              max=\"100\"\r\n              class=\"form-control\"\r\n              formControlName=\"tangoFrom\"\r\n              placeholder=\"1 to 100%\" />\r\n          </ng-template>\r\n        </div>\r\n      </div>\r\n<div\r\n  class=\"mb-3 position-relative\"\r\n  *ngIf=\"showReviewedBy\"\r\n  (click)=\"$event.stopPropagation()\"\r\n>\r\n  <label class=\"form-label\">Reviewed by</label>\r\n\r\n  <!-- Trigger / display -->\r\n  <div\r\n    class=\"border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n    (click)=\"toggleReviewerDropdown($event)\"\r\n  >\r\n    <!-- 0 selected -->\r\n    <span *ngIf=\"!selectedReviewedBy.length\">\r\n      Select\r\n    </span>\r\n\r\n    <!-- 1 selected -->\r\n    <span *ngIf=\"selectedReviewedBy.length === 1\">\r\n      {{ selectedReviewedBy[0] }}\r\n    </span>\r\n\r\n    <!-- >1 selected -->\r\n    <span\r\n      *ngIf=\"selectedReviewedBy.length > 1\"\r\n      class=\"d-flex align-items-center\"\r\n    >\r\n      <span>{{ selectedReviewedBy[0] }}</span>\r\n      <span class=\"badge ms-2\">\r\n        +{{ selectedReviewedBy.length - 1 }}\r\n      </span>\r\n    </span>\r\n  </div>\r\n\r\n  <!-- Dropdown panel -->\r\n  <div\r\n    *ngIf=\"reviewerDropdownOpen\"\r\n    class=\"position-absolute w-100 border rounded bg-white p-1 mt-1\"\r\n    style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n    (click)=\"$event.stopPropagation()\"\r\n  >\r\n    <!-- Select All -->\r\n    <div class=\"form-check px-0 mb-2 d-flex align-items-center\">\r\n      <input\r\n        type=\"checkbox\"\r\n        class=\"form-check-input\"\r\n        id=\"reviewer-all\"\r\n        [checked]=\"areAllReviewersSelected()\"\r\n        (change)=\"onReviewerSelectAll($event)\"\r\n      />\r\n      <label for=\"reviewer-all\" class=\"form-check-label ms-2\">\r\n        Select All\r\n      </label>\r\n    </div>\r\n\r\n    <!-- Options -->\r\n    <div\r\n      class=\"form-check mb-1 px-0 d-flex align-items-center\"\r\n      *ngFor=\"let u of userList\"\r\n    >\r\n      <input\r\n        class=\"form-check-input\"\r\n        type=\"checkbox\"\r\n        [id]=\"'rev-' + u.email\"\r\n        [checked]=\"isReviewerSelected(u.email)\"\r\n        (change)=\"onReviewerChange($event, u.email)\"\r\n      />\r\n      <label class=\"form-check-label ms-2\" [for]=\"'rev-' + u.email\">\r\n        {{ u.email }}\r\n      </label>\r\n    </div>\r\n  </div>\r\n</div>\r\n\r\n\r\n      <!-- Approved By -->\r\n<div\r\n  class=\"mb-3 position-relative\"\r\n  *ngIf=\"showApprovedBy\"\r\n  (click)=\"$event.stopPropagation()\"\r\n>\r\n  <label class=\"form-label\">Approved by</label>\r\n\r\n  <!-- Trigger / display -->\r\n  <div\r\n    class=\"border rounded px-3 py-2 d-flex justify-content-between align-items-center\"\r\n    (click)=\"toggleApproverDropdown($event)\"\r\n  >\r\n    <!-- 0 selected -->\r\n    <span *ngIf=\"!selectedApprovedBy.length\">\r\n      Select\r\n    </span>\r\n\r\n    <!-- 1 selected -->\r\n    <span *ngIf=\"selectedApprovedBy.length === 1\">\r\n      {{ selectedApprovedBy[0] }}\r\n    </span>\r\n\r\n    <!-- >1 selected -->\r\n    <span\r\n      *ngIf=\"selectedApprovedBy.length > 1\"\r\n      class=\"d-flex align-items-center\"\r\n    >\r\n      <span>{{ selectedApprovedBy[0] }}</span>\r\n      <span class=\"badge ms-2\">\r\n        +{{ selectedApprovedBy.length - 1 }}\r\n      </span>\r\n    </span>\r\n\r\n  </div>\r\n\r\n  <!-- Dropdown panel -->\r\n  <div\r\n    *ngIf=\"approverDropdownOpen\"\r\n    class=\"position-absolute w-100 border rounded bg-white p-2 mt-1\"\r\n    style=\"z-index: 1000; max-height: 220px; overflow-y: auto;\"\r\n    (click)=\"$event.stopPropagation()\"\r\n  >\r\n    <!-- Select All -->\r\n    <div class=\"form-check px-0 mb-2 d-flex align-items-center\">\r\n      <input\r\n        type=\"checkbox\"\r\n        class=\"form-check-input\"\r\n        id=\"approver-all\"\r\n        [checked]=\"areAllApproversSelected()\"\r\n        (change)=\"onApproverSelectAll($event)\"\r\n      />\r\n      <label for=\"approver-all\" class=\"form-check-label ms-2\">\r\n        Select All\r\n      </label>\r\n    </div>\r\n\r\n    <!-- Options -->\r\n    <div\r\n      class=\"form-check mb-1 px-0 d-flex align-items-center\"\r\n      *ngFor=\"let u of userList\"\r\n    >\r\n      <input\r\n        class=\"form-check-input\"\r\n        type=\"checkbox\"\r\n        [id]=\"'app-' + u.email\"\r\n        [checked]=\"isApproverSelected(u.email)\"\r\n        (change)=\"onApproverChange($event, u.email)\"\r\n      />\r\n      <label class=\"form-check-label ms-2\" [for]=\"'app-' + u.email\">\r\n        {{ u.email }}\r\n      </label>\r\n    </div>\r\n  </div>\r\n</div>\r\n\r\n\r\n      <!-- Footer buttons -->\r\n      <div class=\"d-flex justify-content-between w-100 mt-4\">\r\n        <button type=\"button\" class=\"btn btn-outline w-50 me-1\" (click)=\"onReset()\">\r\n          Reset\r\n        </button>\r\n        <button type=\"button\" class=\"btn btn-primary w-50 ms-1\" (click)=\"onApply()\">\r\n          Apply\r\n        </button>\r\n      </div>\r\n    </form>\r\n  </div>\r\n</div>\r\n"]}
|