tango-app-ui-manage-tickets 3.7.0-beta.45 → 3.7.0-beta.46
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 +120 -0
- package/esm2022/lib/components/ticket-footfall-new/ticket-footfall-new.component.mjs +15 -3
- package/esm2022/lib/tango-manage-tickets.module.mjs +5 -2
- package/fesm2022/tango-app-ui-manage-tickets.mjs +160 -31
- package/fesm2022/tango-app-ui-manage-tickets.mjs.map +1 -1
- package/lib/components/ticket-filter-panel/ticket-filter-panel.component.d.ts +27 -0
- package/lib/components/ticket-footfall-new/ticket-footfall-new.component.d.ts +3 -1
- package/lib/tango-manage-tickets.module.d.ts +10 -9
- package/package.json +1 -1
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
import { Component, EventEmitter, HostListener, Output } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "@angular/forms";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
5
|
+
export class TicketFilterPanelComponent {
|
|
6
|
+
fb;
|
|
7
|
+
eRef;
|
|
8
|
+
apply = new EventEmitter();
|
|
9
|
+
panelClosed = new EventEmitter(); // ✅ not `close`
|
|
10
|
+
onDocumentClick(event) {
|
|
11
|
+
if (!this.isOpen)
|
|
12
|
+
return;
|
|
13
|
+
const clickedInside = this.eRef.nativeElement.contains(event.target);
|
|
14
|
+
if (!clickedInside) {
|
|
15
|
+
this.close();
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
isOpen = false; // controls visibility
|
|
19
|
+
// called from parent via #filterPanel.open()
|
|
20
|
+
open() {
|
|
21
|
+
this.isOpen = true;
|
|
22
|
+
}
|
|
23
|
+
// called from close button inside the panel
|
|
24
|
+
close() {
|
|
25
|
+
this.isOpen = false;
|
|
26
|
+
this.panelClosed.emit();
|
|
27
|
+
}
|
|
28
|
+
showPanel = false; // you can also control this from parent
|
|
29
|
+
statusOptions = [
|
|
30
|
+
'Open',
|
|
31
|
+
'In-Progress',
|
|
32
|
+
'Close',
|
|
33
|
+
'Under tango review',
|
|
34
|
+
'Tango review done',
|
|
35
|
+
'Expired',
|
|
36
|
+
];
|
|
37
|
+
conditionOptions = [
|
|
38
|
+
{ value: 'gt', label: '> Greater than' },
|
|
39
|
+
{ value: 'lt', label: '< Lesser than' },
|
|
40
|
+
{ value: 'gte', label: '>= Greater than or equal to' },
|
|
41
|
+
{ value: 'lte', label: '<= Lesser than or equal to' },
|
|
42
|
+
{ value: 'between', label: 'Between' },
|
|
43
|
+
];
|
|
44
|
+
filterForm;
|
|
45
|
+
constructor(fb, eRef) {
|
|
46
|
+
this.fb = fb;
|
|
47
|
+
this.eRef = eRef;
|
|
48
|
+
this.filterForm = this.fb.group({
|
|
49
|
+
status: ['Open'],
|
|
50
|
+
reviewerCondition: ['gte'],
|
|
51
|
+
reviewerFrom: [85],
|
|
52
|
+
reviewerTo: [null],
|
|
53
|
+
approverCondition: [null],
|
|
54
|
+
approverFrom: [null],
|
|
55
|
+
approverTo: [null],
|
|
56
|
+
tangoCondition: [null],
|
|
57
|
+
tangoFrom: [null],
|
|
58
|
+
tangoTo: [null],
|
|
59
|
+
approvedBy: [null],
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
isBetween(controlName) {
|
|
63
|
+
return this.filterForm.get(controlName)?.value === 'between';
|
|
64
|
+
}
|
|
65
|
+
onReset() {
|
|
66
|
+
this.filterForm.reset({
|
|
67
|
+
status: null,
|
|
68
|
+
reviewerCondition: null,
|
|
69
|
+
reviewerFrom: null,
|
|
70
|
+
reviewerTo: null,
|
|
71
|
+
approverCondition: null,
|
|
72
|
+
approverFrom: null,
|
|
73
|
+
approverTo: null,
|
|
74
|
+
tangoCondition: null,
|
|
75
|
+
tangoFrom: null,
|
|
76
|
+
tangoTo: null,
|
|
77
|
+
approvedBy: null,
|
|
78
|
+
});
|
|
79
|
+
this.close();
|
|
80
|
+
}
|
|
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
|
+
mapAccuracy(condition, from, to) {
|
|
97
|
+
if (!condition || from == null)
|
|
98
|
+
return null;
|
|
99
|
+
if (condition === 'between') {
|
|
100
|
+
if (to == null)
|
|
101
|
+
return null;
|
|
102
|
+
return { condition, from, to };
|
|
103
|
+
}
|
|
104
|
+
return { condition, value: from };
|
|
105
|
+
}
|
|
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"] }] });
|
|
108
|
+
}
|
|
109
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TicketFilterPanelComponent, decorators: [{
|
|
110
|
+
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: [{
|
|
113
|
+
type: Output
|
|
114
|
+
}], panelClosed: [{
|
|
115
|
+
type: Output
|
|
116
|
+
}], onDocumentClick: [{
|
|
117
|
+
type: HostListener,
|
|
118
|
+
args: ['document:click', ['$event']]
|
|
119
|
+
}] } });
|
|
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"]}
|