tango-app-ui-analyse-zone 3.0.0-alpha.0 → 3.0.0-alpha.2
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/overallcards/overallcards.component.mjs +146 -0
- package/esm2022/lib/components/services/zone.service.mjs +90 -0
- package/esm2022/lib/components/tango-analyse-zone/tango-analyse-zone.component.mjs +45 -5
- package/esm2022/lib/components/top-performing-zones/top-performing-zones.component.mjs +247 -0
- package/esm2022/lib/components/zone-concentration/zone-concentration.component.mjs +234 -0
- package/esm2022/lib/components/zone-summary-table/zone-summary-table.component.mjs +167 -0
- package/esm2022/lib/tango-analyse-zone-routing.module.mjs +8 -2
- package/esm2022/lib/tango-analyse-zone.module.mjs +39 -6
- package/fesm2022/tango-app-ui-analyse-zone.mjs +928 -13
- package/fesm2022/tango-app-ui-analyse-zone.mjs.map +1 -1
- package/lib/components/overallcards/overallcards.component.d.ts +25 -0
- package/lib/components/services/zone.service.d.ts +30 -0
- package/lib/components/tango-analyse-zone/tango-analyse-zone.component.d.ts +12 -1
- package/lib/components/top-performing-zones/top-performing-zones.component.d.ts +39 -0
- package/lib/components/zone-concentration/zone-concentration.component.d.ts +55 -0
- package/lib/components/zone-summary-table/zone-summary-table.component.d.ts +44 -0
- package/lib/tango-analyse-zone.module.d.ts +11 -3
- package/package.json +1 -1
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import * as am5 from '@amcharts/amcharts5';
|
|
3
|
+
import * as am5percent from '@amcharts/amcharts5/percent';
|
|
4
|
+
import { takeUntil } from 'rxjs/operators';
|
|
5
|
+
import { Subject, debounceTime } from 'rxjs';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "../services/zone.service";
|
|
8
|
+
import * as i2 from "tango-app-ui-global";
|
|
9
|
+
import * as i3 from "@angular/common";
|
|
10
|
+
import * as i4 from "@ng-bootstrap/ng-bootstrap";
|
|
11
|
+
export class OverallcardsComponent {
|
|
12
|
+
zone;
|
|
13
|
+
ZoneService;
|
|
14
|
+
changeDetector;
|
|
15
|
+
gs;
|
|
16
|
+
cardData = {};
|
|
17
|
+
genderAnalysis = [];
|
|
18
|
+
ageAnalysis = [];
|
|
19
|
+
cardDataLoading = true;
|
|
20
|
+
cardNoData = false;
|
|
21
|
+
headerData;
|
|
22
|
+
genderroot;
|
|
23
|
+
destroy$ = new Subject();
|
|
24
|
+
constructor(zone, ZoneService, changeDetector, gs) {
|
|
25
|
+
this.zone = zone;
|
|
26
|
+
this.ZoneService = ZoneService;
|
|
27
|
+
this.changeDetector = changeDetector;
|
|
28
|
+
this.gs = gs;
|
|
29
|
+
}
|
|
30
|
+
ngOnInit() {
|
|
31
|
+
this.gs.dataRangeValue
|
|
32
|
+
.pipe(takeUntil(this.destroy$), debounceTime(300))
|
|
33
|
+
.subscribe((data) => {
|
|
34
|
+
this.headerData = data;
|
|
35
|
+
this.getCardData();
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
ngOnDestroy() {
|
|
39
|
+
// this.zone.runOutsideAngular(() => {
|
|
40
|
+
if (this.genderroot) {
|
|
41
|
+
this.genderroot.dispose();
|
|
42
|
+
}
|
|
43
|
+
// });
|
|
44
|
+
this.destroy$.next();
|
|
45
|
+
this.destroy$.complete();
|
|
46
|
+
}
|
|
47
|
+
genderchart() {
|
|
48
|
+
// setTimeout(() => {
|
|
49
|
+
const chartDiv = document.getElementById("genderchartdiv");
|
|
50
|
+
if (!chartDiv) {
|
|
51
|
+
console.error("Could not find HTML element with id `genderchartdiv`");
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
if (this.genderroot) {
|
|
55
|
+
this.genderroot.dispose();
|
|
56
|
+
}
|
|
57
|
+
this.genderroot = am5.Root.new('genderchartdiv');
|
|
58
|
+
const chart = this.genderroot.container.children.push(am5percent.PieChart.new(this.genderroot, {
|
|
59
|
+
endAngle: 270,
|
|
60
|
+
innerRadius: am5.percent(60)
|
|
61
|
+
}));
|
|
62
|
+
const series = chart.series.push(am5percent.PieSeries.new(this.genderroot, {
|
|
63
|
+
valueField: 'value',
|
|
64
|
+
categoryField: 'category',
|
|
65
|
+
endAngle: 270,
|
|
66
|
+
alignLabels: false
|
|
67
|
+
}));
|
|
68
|
+
series.slices.template.setAll({
|
|
69
|
+
cornerRadius: 8
|
|
70
|
+
});
|
|
71
|
+
series.states.create('hidden', {
|
|
72
|
+
endAngle: -90
|
|
73
|
+
});
|
|
74
|
+
series.labels.template.setAll({
|
|
75
|
+
textType: 'circular',
|
|
76
|
+
visible: false
|
|
77
|
+
});
|
|
78
|
+
series.data.setAll(this.genderAnalysis);
|
|
79
|
+
const legend = this.genderroot.container.children.push(am5.Legend.new(this.genderroot, {
|
|
80
|
+
x: am5.percent(50),
|
|
81
|
+
centerX: am5.percent(50),
|
|
82
|
+
y: am5.percent(95),
|
|
83
|
+
layout: this.genderroot.horizontalLayout,
|
|
84
|
+
}));
|
|
85
|
+
legend.labels.template.setAll({
|
|
86
|
+
text: "{category}: {value}%",
|
|
87
|
+
fill: am5.color("#464E5F"),
|
|
88
|
+
fontFamily: "Inter",
|
|
89
|
+
fontSize: "16px",
|
|
90
|
+
fontWeight: "400",
|
|
91
|
+
});
|
|
92
|
+
legend.valueLabels.template.setAll({
|
|
93
|
+
text: "{value}%",
|
|
94
|
+
fontWeight: "700",
|
|
95
|
+
fontFamily: "Inter",
|
|
96
|
+
fontSize: "14px",
|
|
97
|
+
fill: am5.color("#464E5F"),
|
|
98
|
+
});
|
|
99
|
+
legend.data.setAll(series.dataItems);
|
|
100
|
+
// Animation
|
|
101
|
+
series.appear(1000, 100);
|
|
102
|
+
// }, 100);
|
|
103
|
+
}
|
|
104
|
+
getCardData() {
|
|
105
|
+
this.cardDataLoading = true;
|
|
106
|
+
this.cardNoData = true;
|
|
107
|
+
const requestData = {
|
|
108
|
+
fromDate: this.headerData.date.startDate,
|
|
109
|
+
toDate: this.headerData.date.endDate,
|
|
110
|
+
storeId: this.headerData.stores.filter((store) => store.checked).map((store) => store.storeId),
|
|
111
|
+
// storeId: ["11-253", "11-511","11-1176"],
|
|
112
|
+
clientId: this.headerData.client,
|
|
113
|
+
};
|
|
114
|
+
this.ZoneService.getCardData(requestData).pipe(takeUntil(this.destroy$)).subscribe((response) => {
|
|
115
|
+
this.cardDataLoading = false;
|
|
116
|
+
if (response?.code === 200 && response?.status === "success") {
|
|
117
|
+
this.cardData = response.data.card;
|
|
118
|
+
this.genderAnalysis = response.data.genderAnalysis;
|
|
119
|
+
this.genderAnalysis.forEach((item) => {
|
|
120
|
+
item.value = Math.round(item.value);
|
|
121
|
+
});
|
|
122
|
+
this.ageAnalysis = response.data.ageAnalysis;
|
|
123
|
+
setTimeout(() => {
|
|
124
|
+
this.genderchart(); // Call this method after data is set
|
|
125
|
+
}, 200);
|
|
126
|
+
this.cardDataLoading = false;
|
|
127
|
+
this.cardNoData = false;
|
|
128
|
+
}
|
|
129
|
+
else {
|
|
130
|
+
this.cardData = [];
|
|
131
|
+
this.cardDataLoading = false;
|
|
132
|
+
this.cardNoData = true;
|
|
133
|
+
}
|
|
134
|
+
}, error => {
|
|
135
|
+
this.cardData = [];
|
|
136
|
+
this.changeDetector.detectChanges();
|
|
137
|
+
});
|
|
138
|
+
}
|
|
139
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: OverallcardsComponent, deps: [{ token: i0.NgZone }, { token: i1.ZoneService }, { token: i0.ChangeDetectorRef }, { token: i2.GlobalStateService }], target: i0.ɵɵFactoryTarget.Component });
|
|
140
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: OverallcardsComponent, selector: "lib-overallcards", ngImport: i0, template: "<div class=\"row\">\r\n <div class=\"col-lg-4\">\r\n <div class=\"card\">\r\n <div class=\"card-header py-3\">\r\n <h3 class=\"card-title align-items-start flex-column\">\r\n <span class=\"subtext\">Overall Zone Footfall</span>\r\n </h3>\r\n <div class=\"card-toolbar\">\r\n <div class=\"headcount\">{{ cardData.oveallZoneFootfall?.count || '--' }}</div>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"row my-2 justify-content-evenly\">\r\n <div class=\"col-6 ratecards\">\r\n <div class=\"headcount\">{{ cardData.bounced?.rate?.toFixed(0) || '--' }}%</div>\r\n <div class=\"subtext mt-3\">Bounced Footfall Rate</div>\r\n </div>\r\n <div class=\"col-6 ratecards\">\r\n <div class=\"headcount\">{{ cardData.engagers?.rate?.toFixed(0) || '--' }}%</div>\r\n <div class=\"subtext mt-3\">Engagers Rate</div>\r\n </div>\r\n </div>\r\n <div class=\"row my-2 justify-content-evenly\">\r\n <div class=\"col-6 ratecards\">\r\n <div class=\"headcount\">{{ cardData.avgDwellTime || '--' }} Mins</div>\r\n <div class=\"subtext mt-3\">Average Dwell Time</div>\r\n </div>\r\n <div class=\"col-6 ratecards\">\r\n <div class=\"headcount\">{{ cardData.concentration?.rate?.toFixed(0) || '--' }}%</div>\r\n <div class=\"subtext mt-3\">Concentration Rate</div>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"col-4\">\r\n <div class=\"card\">\r\n <div class=\"headtext p-3\">Gender Analysis</div>\r\n <div class=\"card-body p-0\">\r\n <div *ngIf=\"!cardDataLoading && !cardNoData\" id=\"genderchartdiv\"></div>\r\n <ng-container *ngIf=\"cardDataLoading\">\r\n <div class=\"row loader d-flex justify-content-center align-items-center\">\r\n <div class=\"shimmer\">\r\n <div class=\"wrapper\">\r\n <div class=\"stroke animate title\"></div>\r\n <div class=\"stroke animate link\"></div>\r\n <div class=\"stroke animate description\"></div>\r\n </div>\r\n </div>\r\n <div class=\"shimmer\">\r\n <div class=\"wrapper\">\r\n <div class=\"stroke animate title\"></div>\r\n <div class=\"stroke animate link\"></div>\r\n <div class=\"stroke animate description\"></div>\r\n </div>\r\n </div>\r\n \r\n </div>\r\n </ng-container>\r\n <ng-container *ngIf=\"cardNoData && !cardDataLoading\">\r\n <div class=\"row\">\r\n <div class=\"col-lg-12\">\r\n <div\r\n class=\"card-body my-8 d-flex justify-content-center align-items-center flex-column\">\r\n <img class=\"img-src\" src=\"./assets/tango/Icons/Nodata1.svg\" alt=\"\">\r\n <div class=\"nodatamaintext mt-3\">No data found</div>\r\n <div class=\"nodatasubtext\">There is no result for Gender Analysis</div>\r\n </div>\r\n </div>\r\n </div>\r\n </ng-container>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"col-4\">\r\n <div class=\"card\">\r\n <div class=\"headtext p-3\">Age Analysis</div>\r\n <div class=\"card-body p-0\">\r\n <div class=\"mx-3\">\r\n <table class=\"table bottom-border\">\r\n <thead>\r\n <tr>\r\n <th>Age Range</th>\r\n <th>Percentage of FF</th>\r\n </tr>\r\n </thead>\r\n <tbody *ngIf=\"!cardDataLoading && !cardNoData\">\r\n <tr *ngFor=\"let item of ageAnalysis\">\r\n <td>\r\n <div class=\"table-title subscribedtext\">\r\n {{ item.category }}\r\n </div>\r\n </td>\r\n <td>\r\n <div class=\"row\">\r\n <div class=\"col-6 mt-2\">\r\n <ngb-progressbar class=\"mb-3\" height=\"10px\" type=\"primary\" [value]=\"item.value\" />\r\n </div>\r\n <div class=\"col-6\">\r\n {{ item.value?.toFixed(0) || '--'}}%\r\n \r\n </div>\r\n </div>\r\n </td>\r\n </tr>\r\n </tbody>\r\n </table>\r\n <ng-container *ngIf=\"cardDataLoading\">\r\n <div class=\"row loader d-flex justify-content-center align-items-center\">\r\n <div class=\"shimmer my-17\">\r\n <div class=\"wrapper\">\r\n <div class=\"stroke animate title\"></div>\r\n <div class=\"stroke animate link\"></div>\r\n <div class=\"stroke animate description\"></div>\r\n </div>\r\n </div>\r\n <!-- <div class=\"shimmer\">\r\n <div class=\"wrapper\">\r\n <div class=\"stroke animate title\"></div>\r\n <div class=\"stroke animate link\"></div>\r\n <div class=\"stroke animate description\"></div>\r\n </div>\r\n </div> -->\r\n \r\n </div>\r\n </ng-container>\r\n <ng-container *ngIf=\"cardNoData && !cardDataLoading\">\r\n <div class=\"row\">\r\n <div class=\"col-lg-12\">\r\n <div\r\n class=\"card-body d-flex justify-content-center align-items-center flex-column\">\r\n <img class=\"img-src\" src=\"./assets/tango/Icons/Nodata1.svg\" alt=\"\">\r\n <div class=\"nodatamaintext mt-3\">No data found</div>\r\n <div class=\"nodatasubtext\">There is no result for Age Analysis</div>\r\n </div>\r\n </div>\r\n </div>\r\n </ng-container>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n", styles: [".subtext{color:var(--Gray-800, #1D2939);font-size:14px;font-weight:500;line-height:20px}.headcount{color:var(--Gray-900, #101828);font-size:20px;font-weight:600;line-height:30px}.headtext{color:var(--Gray-700, #344054);font-size:18px;font-weight:600;line-height:28px}.ratecards{padding:30px;width:47%;border-radius:12px;background:var(--White, #FFF)}.primarybar{background:#00a3ff}.card{border-radius:12px}#genderchartdiv{width:100%;height:241px;margin-bottom:6%}table th,table td{height:37.5px!important;padding:0 22px!important;align-items:center;gap:12px;align-self:stretch}table tr{vertical-align:middle}.nodatamaintext{font-family:Inter;font-size:16px;font-weight:600;line-height:24px;text-align:center;color:#101828}.nodatasubtext{font-family:Inter;font-size:14px;font-weight:400;line-height:20px;text-align:center;color:#667085}\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: "component", type: i4.NgbProgressbar, selector: "ngb-progressbar", inputs: ["max", "animated", "ariaLabel", "striped", "showValue", "textType", "type", "value", "height"] }] });
|
|
141
|
+
}
|
|
142
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: OverallcardsComponent, decorators: [{
|
|
143
|
+
type: Component,
|
|
144
|
+
args: [{ selector: 'lib-overallcards', template: "<div class=\"row\">\r\n <div class=\"col-lg-4\">\r\n <div class=\"card\">\r\n <div class=\"card-header py-3\">\r\n <h3 class=\"card-title align-items-start flex-column\">\r\n <span class=\"subtext\">Overall Zone Footfall</span>\r\n </h3>\r\n <div class=\"card-toolbar\">\r\n <div class=\"headcount\">{{ cardData.oveallZoneFootfall?.count || '--' }}</div>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"row my-2 justify-content-evenly\">\r\n <div class=\"col-6 ratecards\">\r\n <div class=\"headcount\">{{ cardData.bounced?.rate?.toFixed(0) || '--' }}%</div>\r\n <div class=\"subtext mt-3\">Bounced Footfall Rate</div>\r\n </div>\r\n <div class=\"col-6 ratecards\">\r\n <div class=\"headcount\">{{ cardData.engagers?.rate?.toFixed(0) || '--' }}%</div>\r\n <div class=\"subtext mt-3\">Engagers Rate</div>\r\n </div>\r\n </div>\r\n <div class=\"row my-2 justify-content-evenly\">\r\n <div class=\"col-6 ratecards\">\r\n <div class=\"headcount\">{{ cardData.avgDwellTime || '--' }} Mins</div>\r\n <div class=\"subtext mt-3\">Average Dwell Time</div>\r\n </div>\r\n <div class=\"col-6 ratecards\">\r\n <div class=\"headcount\">{{ cardData.concentration?.rate?.toFixed(0) || '--' }}%</div>\r\n <div class=\"subtext mt-3\">Concentration Rate</div>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"col-4\">\r\n <div class=\"card\">\r\n <div class=\"headtext p-3\">Gender Analysis</div>\r\n <div class=\"card-body p-0\">\r\n <div *ngIf=\"!cardDataLoading && !cardNoData\" id=\"genderchartdiv\"></div>\r\n <ng-container *ngIf=\"cardDataLoading\">\r\n <div class=\"row loader d-flex justify-content-center align-items-center\">\r\n <div class=\"shimmer\">\r\n <div class=\"wrapper\">\r\n <div class=\"stroke animate title\"></div>\r\n <div class=\"stroke animate link\"></div>\r\n <div class=\"stroke animate description\"></div>\r\n </div>\r\n </div>\r\n <div class=\"shimmer\">\r\n <div class=\"wrapper\">\r\n <div class=\"stroke animate title\"></div>\r\n <div class=\"stroke animate link\"></div>\r\n <div class=\"stroke animate description\"></div>\r\n </div>\r\n </div>\r\n \r\n </div>\r\n </ng-container>\r\n <ng-container *ngIf=\"cardNoData && !cardDataLoading\">\r\n <div class=\"row\">\r\n <div class=\"col-lg-12\">\r\n <div\r\n class=\"card-body my-8 d-flex justify-content-center align-items-center flex-column\">\r\n <img class=\"img-src\" src=\"./assets/tango/Icons/Nodata1.svg\" alt=\"\">\r\n <div class=\"nodatamaintext mt-3\">No data found</div>\r\n <div class=\"nodatasubtext\">There is no result for Gender Analysis</div>\r\n </div>\r\n </div>\r\n </div>\r\n </ng-container>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"col-4\">\r\n <div class=\"card\">\r\n <div class=\"headtext p-3\">Age Analysis</div>\r\n <div class=\"card-body p-0\">\r\n <div class=\"mx-3\">\r\n <table class=\"table bottom-border\">\r\n <thead>\r\n <tr>\r\n <th>Age Range</th>\r\n <th>Percentage of FF</th>\r\n </tr>\r\n </thead>\r\n <tbody *ngIf=\"!cardDataLoading && !cardNoData\">\r\n <tr *ngFor=\"let item of ageAnalysis\">\r\n <td>\r\n <div class=\"table-title subscribedtext\">\r\n {{ item.category }}\r\n </div>\r\n </td>\r\n <td>\r\n <div class=\"row\">\r\n <div class=\"col-6 mt-2\">\r\n <ngb-progressbar class=\"mb-3\" height=\"10px\" type=\"primary\" [value]=\"item.value\" />\r\n </div>\r\n <div class=\"col-6\">\r\n {{ item.value?.toFixed(0) || '--'}}%\r\n \r\n </div>\r\n </div>\r\n </td>\r\n </tr>\r\n </tbody>\r\n </table>\r\n <ng-container *ngIf=\"cardDataLoading\">\r\n <div class=\"row loader d-flex justify-content-center align-items-center\">\r\n <div class=\"shimmer my-17\">\r\n <div class=\"wrapper\">\r\n <div class=\"stroke animate title\"></div>\r\n <div class=\"stroke animate link\"></div>\r\n <div class=\"stroke animate description\"></div>\r\n </div>\r\n </div>\r\n <!-- <div class=\"shimmer\">\r\n <div class=\"wrapper\">\r\n <div class=\"stroke animate title\"></div>\r\n <div class=\"stroke animate link\"></div>\r\n <div class=\"stroke animate description\"></div>\r\n </div>\r\n </div> -->\r\n \r\n </div>\r\n </ng-container>\r\n <ng-container *ngIf=\"cardNoData && !cardDataLoading\">\r\n <div class=\"row\">\r\n <div class=\"col-lg-12\">\r\n <div\r\n class=\"card-body d-flex justify-content-center align-items-center flex-column\">\r\n <img class=\"img-src\" src=\"./assets/tango/Icons/Nodata1.svg\" alt=\"\">\r\n <div class=\"nodatamaintext mt-3\">No data found</div>\r\n <div class=\"nodatasubtext\">There is no result for Age Analysis</div>\r\n </div>\r\n </div>\r\n </div>\r\n </ng-container>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n", styles: [".subtext{color:var(--Gray-800, #1D2939);font-size:14px;font-weight:500;line-height:20px}.headcount{color:var(--Gray-900, #101828);font-size:20px;font-weight:600;line-height:30px}.headtext{color:var(--Gray-700, #344054);font-size:18px;font-weight:600;line-height:28px}.ratecards{padding:30px;width:47%;border-radius:12px;background:var(--White, #FFF)}.primarybar{background:#00a3ff}.card{border-radius:12px}#genderchartdiv{width:100%;height:241px;margin-bottom:6%}table th,table td{height:37.5px!important;padding:0 22px!important;align-items:center;gap:12px;align-self:stretch}table tr{vertical-align:middle}.nodatamaintext{font-family:Inter;font-size:16px;font-weight:600;line-height:24px;text-align:center;color:#101828}.nodatasubtext{font-family:Inter;font-size:14px;font-weight:400;line-height:20px;text-align:center;color:#667085}\n"] }]
|
|
145
|
+
}], ctorParameters: () => [{ type: i0.NgZone }, { type: i1.ZoneService }, { type: i0.ChangeDetectorRef }, { type: i2.GlobalStateService }] });
|
|
146
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"overallcards.component.js","sourceRoot":"","sources":["../../../../../../projects/tango-analyse-zone/src/lib/components/overallcards/overallcards.component.ts","../../../../../../projects/tango-analyse-zone/src/lib/components/overallcards/overallcards.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, Component, NgZone } from '@angular/core';\r\nimport * as am5 from '@amcharts/amcharts5';\r\nimport * as am5percent from '@amcharts/amcharts5/percent';\r\nimport * as am5themes_Animated from '@amcharts/amcharts5/themes/Animated';\r\nimport { takeUntil } from 'rxjs/operators';\r\nimport { Subject, debounceTime} from 'rxjs';\r\nimport { ZoneService } from '../services/zone.service';\r\nimport { GlobalStateService } from 'tango-app-ui-global';\r\n\r\n@Component({\r\n  selector: 'lib-overallcards',\r\n  templateUrl: './overallcards.component.html',\r\n  styleUrl: './overallcards.component.scss',\r\n})\r\nexport class OverallcardsComponent {\r\n  cardData: any = {};\r\n  genderAnalysis: any[] = [];\r\n  ageAnalysis: any[] = [];\r\n  cardDataLoading: boolean = true;\r\n  cardNoData: boolean = false;\r\n  headerData: any;\r\n  private genderroot!: am5.Root;\r\n  private destroy$ = new Subject<void>();\r\n  constructor(private zone: NgZone,private ZoneService: ZoneService,private changeDetector: ChangeDetectorRef,public gs: GlobalStateService,) {}\r\n\r\n  ngOnInit(): void {\r\n    this.gs.dataRangeValue\r\n    .pipe(takeUntil(this.destroy$), debounceTime(300))\r\n    .subscribe((data: any) => {\r\n        this.headerData = data;\r\n        this.getCardData();\r\n})\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    // this.zone.runOutsideAngular(() => {\r\n      if (this.genderroot) {\r\n        this.genderroot.dispose();\r\n      }\r\n    // });\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n  }\r\n\r\n  genderchart() {\r\n      // setTimeout(() => {\r\n        const chartDiv = document.getElementById(\"genderchartdiv\");\r\n        if (!chartDiv) {\r\n          console.error(\"Could not find HTML element with id `genderchartdiv`\");\r\n          return;\r\n        }\r\n    \r\n        if (this.genderroot) {\r\n          this.genderroot.dispose();\r\n        }\r\n        this.genderroot = am5.Root.new('genderchartdiv');\r\n    \r\n        const chart = this.genderroot.container.children.push(\r\n            am5percent.PieChart.new(this.genderroot, {\r\n                endAngle: 270,\r\n                innerRadius: am5.percent(60)\r\n            })\r\n        );\r\n  \r\n        const series = chart.series.push(\r\n            am5percent.PieSeries.new(this.genderroot, {\r\n                valueField: 'value',\r\n                categoryField: 'category',\r\n                endAngle: 270,\r\n                alignLabels: false\r\n            })\r\n        );\r\n  \r\n        series.slices.template.setAll({\r\n            cornerRadius: 8\r\n        });\r\n  \r\n        series.states.create('hidden', {\r\n            endAngle: -90\r\n        });\r\n  \r\n        series.labels.template.setAll({\r\n            textType: 'circular',\r\n            visible: false\r\n        });\r\n\r\n        series.data.setAll(this.genderAnalysis);\r\n        const legend = this.genderroot.container.children.push(\r\n            am5.Legend.new(this.genderroot, {\r\n                x: am5.percent(50),\r\n                centerX: am5.percent(50),\r\n                y: am5.percent(95),\r\n                layout: this.genderroot.horizontalLayout,\r\n            })\r\n        );\r\n\r\n        legend.labels.template.setAll({\r\n          text: \"{category}: {value}%\",\r\n          fill: am5.color(\"#464E5F\"),\r\n          fontFamily: \"Inter\",\r\n          fontSize: \"16px\",\r\n          fontWeight: \"400\",\r\n          \r\n        });\r\n        \r\n        legend.valueLabels.template.setAll({\r\n          text: \"{value}%\",\r\n          fontWeight: \"700\",\r\n          fontFamily: \"Inter\",\r\n          fontSize: \"14px\",\r\n          fill: am5.color(\"#464E5F\"),\r\n        });\r\n  \r\n        legend.data.setAll(series.dataItems);\r\n  \r\n        // Animation\r\n        series.appear(1000, 100);\r\n    // }, 100);\r\n  }\r\n  \r\n\r\n  getCardData(): void {\r\n    this.cardDataLoading = true;\r\n    this.cardNoData = true;\r\n    const requestData = {\r\n      fromDate: this.headerData.date.startDate,\r\n      toDate: this.headerData.date.endDate,\r\n      storeId: this.headerData.stores.filter((store:any) => store.checked).map((store:any) => store.storeId),\r\n      // storeId: [\"11-253\", \"11-511\",\"11-1176\"],\r\n      clientId:this.headerData.client,\r\n    };\r\n\r\n    this.ZoneService.getCardData(requestData).pipe(takeUntil(this.destroy$)).subscribe((response: any) => {\r\n      this.cardDataLoading = false;\r\n        if (response?.code === 200 && response?.status === \"success\") {\r\n          this.cardData = response.data.card;\r\n          this.genderAnalysis = response.data.genderAnalysis;\r\n          this.genderAnalysis.forEach((item:any) => {\r\n            item.value = Math.round(item.value)\r\n          })\r\n          this.ageAnalysis = response.data.ageAnalysis;\r\n          setTimeout(() => {\r\n            this.genderchart(); // Call this method after data is set\r\n          }, 200);\r\n          \r\n          this.cardDataLoading = false;\r\n          this.cardNoData = false;\r\n        } else {\r\n          this.cardData = []\r\n          this.cardDataLoading = false;\r\n          this.cardNoData = true;\r\n        }\r\n      },\r\n      error => {\r\n        this.cardData = []\r\n      this.changeDetector.detectChanges()\r\n      }\r\n    );\r\n  }\r\n  \r\n  // roundof(data:any) {\r\n  //   return data ? Math.round(parseInt(data)) : '--';\r\n  // }\r\n}\r\n","<div class=\"row\">\r\n    <div class=\"col-lg-4\">\r\n        <div class=\"card\">\r\n            <div class=\"card-header py-3\">\r\n                <h3 class=\"card-title align-items-start flex-column\">\r\n                    <span class=\"subtext\">Overall Zone Footfall</span>\r\n                </h3>\r\n                <div class=\"card-toolbar\">\r\n                    <div class=\"headcount\">{{ cardData.oveallZoneFootfall?.count || '--'  }}</div>\r\n                </div>\r\n            </div>\r\n        </div>\r\n        <div class=\"row my-2 justify-content-evenly\">\r\n            <div class=\"col-6 ratecards\">\r\n                <div class=\"headcount\">{{ cardData.bounced?.rate?.toFixed(0) || '--'  }}%</div>\r\n                <div class=\"subtext mt-3\">Bounced Footfall Rate</div>\r\n            </div>\r\n            <div class=\"col-6 ratecards\">\r\n                <div class=\"headcount\">{{ cardData.engagers?.rate?.toFixed(0) || '--'  }}%</div>\r\n                <div class=\"subtext mt-3\">Engagers Rate</div>\r\n            </div>\r\n        </div>\r\n        <div class=\"row my-2 justify-content-evenly\">\r\n            <div class=\"col-6 ratecards\">\r\n                <div class=\"headcount\">{{ cardData.avgDwellTime || '--'  }} Mins</div>\r\n                <div class=\"subtext mt-3\">Average Dwell Time</div>\r\n            </div>\r\n            <div class=\"col-6 ratecards\">\r\n                <div class=\"headcount\">{{ cardData.concentration?.rate?.toFixed(0) || '--'  }}%</div>\r\n                <div class=\"subtext mt-3\">Concentration Rate</div>\r\n            </div>\r\n        </div>\r\n    </div>\r\n    <div class=\"col-4\">\r\n        <div class=\"card\">\r\n            <div class=\"headtext p-3\">Gender Analysis</div>\r\n            <div class=\"card-body p-0\">\r\n                <div *ngIf=\"!cardDataLoading && !cardNoData\" id=\"genderchartdiv\"></div>\r\n                <ng-container *ngIf=\"cardDataLoading\">\r\n                    <div class=\"row loader d-flex justify-content-center align-items-center\">\r\n                        <div class=\"shimmer\">\r\n                            <div class=\"wrapper\">\r\n                                <div class=\"stroke animate title\"></div>\r\n                                <div class=\"stroke animate link\"></div>\r\n                                <div class=\"stroke animate description\"></div>\r\n                            </div>\r\n                        </div>\r\n                        <div class=\"shimmer\">\r\n                            <div class=\"wrapper\">\r\n                                <div class=\"stroke animate title\"></div>\r\n                                <div class=\"stroke animate link\"></div>\r\n                                <div class=\"stroke animate description\"></div>\r\n                            </div>\r\n                        </div>\r\n    \r\n                    </div>\r\n                </ng-container>\r\n                <ng-container *ngIf=\"cardNoData && !cardDataLoading\">\r\n                    <div class=\"row\">\r\n                        <div class=\"col-lg-12\">\r\n                            <div\r\n                                class=\"card-body my-8 d-flex justify-content-center align-items-center flex-column\">\r\n                                <img class=\"img-src\" src=\"./assets/tango/Icons/Nodata1.svg\" alt=\"\">\r\n                                <div class=\"nodatamaintext mt-3\">No data found</div>\r\n                                <div class=\"nodatasubtext\">There is no result for Gender Analysis</div>\r\n                            </div>\r\n                        </div>\r\n                    </div>\r\n                </ng-container>\r\n            </div>\r\n        </div>\r\n    </div>\r\n    <div class=\"col-4\">\r\n        <div class=\"card\">\r\n            <div class=\"headtext p-3\">Age Analysis</div>\r\n            <div class=\"card-body p-0\">\r\n                <div class=\"mx-3\">\r\n                    <table class=\"table bottom-border\">\r\n                        <thead>\r\n                            <tr>\r\n                                <th>Age Range</th>\r\n                                <th>Percentage of FF</th>\r\n                            </tr>\r\n                        </thead>\r\n                        <tbody  *ngIf=\"!cardDataLoading && !cardNoData\">\r\n                            <tr *ngFor=\"let item of ageAnalysis\">\r\n                                <td>\r\n                                    <div class=\"table-title subscribedtext\">\r\n                                        {{ item.category }}\r\n                                    </div>\r\n                                </td>\r\n                                <td>\r\n                                    <div class=\"row\">\r\n                                        <div class=\"col-6 mt-2\">\r\n                                            <ngb-progressbar class=\"mb-3\" height=\"10px\" type=\"primary\" [value]=\"item.value\" />\r\n                                        </div>\r\n                                        <div class=\"col-6\">\r\n                                            {{ item.value?.toFixed(0) || '--'}}%\r\n                                            \r\n                                        </div>\r\n                                    </div>\r\n                                </td>\r\n                            </tr>\r\n                        </tbody>\r\n                    </table>\r\n                    <ng-container *ngIf=\"cardDataLoading\">\r\n                        <div class=\"row loader d-flex justify-content-center align-items-center\">\r\n                            <div class=\"shimmer my-17\">\r\n                                <div class=\"wrapper\">\r\n                                    <div class=\"stroke animate title\"></div>\r\n                                    <div class=\"stroke animate link\"></div>\r\n                                    <div class=\"stroke animate description\"></div>\r\n                                </div>\r\n                            </div>\r\n                            <!-- <div class=\"shimmer\">\r\n                                <div class=\"wrapper\">\r\n                                    <div class=\"stroke animate title\"></div>\r\n                                    <div class=\"stroke animate link\"></div>\r\n                                    <div class=\"stroke animate description\"></div>\r\n                                </div>\r\n                            </div> -->\r\n        \r\n                        </div>\r\n                    </ng-container>\r\n                    <ng-container *ngIf=\"cardNoData && !cardDataLoading\">\r\n                        <div class=\"row\">\r\n                            <div class=\"col-lg-12\">\r\n                                <div\r\n                                    class=\"card-body d-flex justify-content-center align-items-center flex-column\">\r\n                                    <img class=\"img-src\" src=\"./assets/tango/Icons/Nodata1.svg\" alt=\"\">\r\n                                    <div class=\"nodatamaintext mt-3\">No data found</div>\r\n                                    <div class=\"nodatasubtext\">There is no result for Age Analysis</div>\r\n                                </div>\r\n                            </div>\r\n                        </div>\r\n                    </ng-container>\r\n                </div>\r\n            </div>\r\n        </div>\r\n    </div>\r\n</div>\r\n"]}
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import { Injectable, } from '@angular/core';
|
|
2
|
+
import { BehaviorSubject, Subject, takeUntil, throwError } from 'rxjs';
|
|
3
|
+
import * as FileSaver from 'file-saver';
|
|
4
|
+
import * as XLSX from 'xlsx';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@angular/common/http";
|
|
7
|
+
import * as i2 from "tango-app-ui-global";
|
|
8
|
+
const EXCEL_TYPE = 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=UTF-8';
|
|
9
|
+
const EXCEL_EXTENSION = '.xlsx';
|
|
10
|
+
export class ZoneService {
|
|
11
|
+
http;
|
|
12
|
+
gs;
|
|
13
|
+
zoneAnalysisUrl;
|
|
14
|
+
reloadDataSubject = new BehaviorSubject(false);
|
|
15
|
+
reloadData$ = this.reloadDataSubject.asObservable();
|
|
16
|
+
constructor(http, gs) {
|
|
17
|
+
this.http = http;
|
|
18
|
+
this.gs = gs;
|
|
19
|
+
this.gs.environment.pipe(takeUntil(this.destroy$)).subscribe((env) => {
|
|
20
|
+
if (env) {
|
|
21
|
+
this.zoneAnalysisUrl = env.zoneAnalysisUrl;
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
triggerReload() {
|
|
26
|
+
this.reloadDataSubject.next(true);
|
|
27
|
+
}
|
|
28
|
+
exportAsExcelFile(json, excelFileName) {
|
|
29
|
+
const worksheet = XLSX.utils.json_to_sheet(json);
|
|
30
|
+
const workbook = { Sheets: { 'data': worksheet }, SheetNames: ['data'] };
|
|
31
|
+
const excelBuffer = XLSX.write(workbook, { bookType: 'xlsx', type: 'array' });
|
|
32
|
+
this.saveAsExcelFile(excelBuffer, excelFileName);
|
|
33
|
+
}
|
|
34
|
+
saveAsExcelFile(buffer, fileName) {
|
|
35
|
+
const data = new Blob([buffer], {
|
|
36
|
+
type: EXCEL_TYPE
|
|
37
|
+
});
|
|
38
|
+
FileSaver.saveAs(data, fileName + 'list' + EXCEL_EXTENSION);
|
|
39
|
+
}
|
|
40
|
+
saveAsTemplate(buffer, fileName) {
|
|
41
|
+
const data = new Blob([buffer], {
|
|
42
|
+
type: EXCEL_TYPE
|
|
43
|
+
});
|
|
44
|
+
FileSaver.saveAs(data, fileName + EXCEL_EXTENSION);
|
|
45
|
+
}
|
|
46
|
+
destroy$ = new Subject();
|
|
47
|
+
ngOnDestroy() {
|
|
48
|
+
this.destroy$.next(true);
|
|
49
|
+
this.destroy$.complete();
|
|
50
|
+
}
|
|
51
|
+
zoneConcentrationTableData(data) {
|
|
52
|
+
return this.http.post(`${this.zoneAnalysisUrl}/zoneConcentrationSummaryTable_v1`, data);
|
|
53
|
+
}
|
|
54
|
+
getOverallStoreConcentrationData(data) {
|
|
55
|
+
return this.http.post(`${this.zoneAnalysisUrl}/zoneHeatmapDatas_v1`, data);
|
|
56
|
+
}
|
|
57
|
+
getOverallStoreHeatmapDates(data) {
|
|
58
|
+
return this.http.post(`${this.zoneAnalysisUrl}/zoneHeatmapAvailableDates_v1`, data);
|
|
59
|
+
}
|
|
60
|
+
getCardData(data) {
|
|
61
|
+
return this.http.post(`${this.zoneAnalysisUrl}/cardsAgeAnalysis_v1`, data);
|
|
62
|
+
}
|
|
63
|
+
getTopPerformingZonesData(data) {
|
|
64
|
+
return this.http.post(`${this.zoneAnalysisUrl}/topPerformingZones_v1`, data);
|
|
65
|
+
}
|
|
66
|
+
getTopPerformingStoresTableData(data) {
|
|
67
|
+
return this.http.post(`${this.zoneAnalysisUrl}/topPerformingStores_v1`, data);
|
|
68
|
+
}
|
|
69
|
+
getTopPerformingStoresTableExport(data) {
|
|
70
|
+
return this.http.post(`${this.zoneAnalysisUrl}/topPerformingStores_v1`, data, { responseType: 'arraybuffer' });
|
|
71
|
+
}
|
|
72
|
+
getSummaryTableData(data) {
|
|
73
|
+
return this.http.post(`${this.zoneAnalysisUrl}/zoneSummaryTable_v1`, data);
|
|
74
|
+
}
|
|
75
|
+
getSummaryTableExport(data) {
|
|
76
|
+
return this.http.post(`${this.zoneAnalysisUrl}/zoneSummaryTable_v1`, data, { responseType: 'arraybuffer' });
|
|
77
|
+
}
|
|
78
|
+
handleError(error) {
|
|
79
|
+
return throwError(() => new Error("Something bad happened; please try again later"));
|
|
80
|
+
}
|
|
81
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ZoneService, deps: [{ token: i1.HttpClient }, { token: i2.GlobalStateService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
82
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ZoneService, providedIn: 'root' });
|
|
83
|
+
}
|
|
84
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ZoneService, decorators: [{
|
|
85
|
+
type: Injectable,
|
|
86
|
+
args: [{
|
|
87
|
+
providedIn: 'root'
|
|
88
|
+
}]
|
|
89
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.GlobalStateService }] });
|
|
90
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"zone.service.js","sourceRoot":"","sources":["../../../../../../projects/tango-analyse-zone/src/lib/components/services/zone.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpErrorResponse } from '@angular/common/http';\r\nimport { Injectable, } from '@angular/core';\r\nimport { GlobalStateService } from 'tango-app-ui-global';\r\nimport { BehaviorSubject, Observable, Subject, takeUntil, catchError, map,throwError } from 'rxjs';\r\nimport * as FileSaver from 'file-saver';\r\nimport * as XLSX from 'xlsx';\r\n \r\nconst EXCEL_TYPE = 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=UTF-8';\r\nconst EXCEL_EXTENSION = '.xlsx';\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class ZoneService {\r\n  zoneAnalysisUrl:any;\r\n  private reloadDataSubject = new BehaviorSubject<boolean>(false);\r\n  reloadData$ = this.reloadDataSubject.asObservable();\r\n\r\n  constructor(private http: HttpClient, private gs: GlobalStateService) {\r\n    this.gs.environment.pipe(takeUntil(this.destroy$)).subscribe((env) => {\r\n      if (env) {\r\n        this.zoneAnalysisUrl = env.zoneAnalysisUrl;\r\n\r\n      }\r\n    });\r\n  }\r\n  triggerReload() {\r\n    this.reloadDataSubject.next(true);\r\n  }\r\n  public exportAsExcelFile(json: any[], excelFileName: string): void {\r\n    const worksheet: XLSX.WorkSheet = XLSX.utils.json_to_sheet(json);\r\n    const workbook: XLSX.WorkBook = { Sheets: { 'data': worksheet }, SheetNames: ['data'] };\r\n    const excelBuffer: any = XLSX.write(workbook, { bookType: 'xlsx', type: 'array' });\r\n    this.saveAsExcelFile(excelBuffer, excelFileName);\r\n  }\r\n \r\n  public saveAsExcelFile(buffer: any, fileName: string): void {\r\n    const data: Blob = new Blob([buffer], {\r\n      type: EXCEL_TYPE\r\n    });\r\n    FileSaver.saveAs(data, fileName + 'list'+ EXCEL_EXTENSION);\r\n  }\r\n \r\n  public saveAsTemplate(buffer: any, fileName: string): void {\r\n    const data: Blob = new Blob([buffer], {\r\n      type: EXCEL_TYPE\r\n    });\r\n    FileSaver.saveAs(data, fileName + EXCEL_EXTENSION);\r\n  }\r\n  private readonly destroy$ = new Subject();\r\n\r\nngOnDestroy(): void {\r\n  this.destroy$.next(true);\r\n  this.destroy$.complete();\r\n}\r\n\r\n  zoneConcentrationTableData( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/zoneConcentrationSummaryTable_v1`, data)\r\n  }\r\n  getOverallStoreConcentrationData( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/zoneHeatmapDatas_v1`, data)\r\n  }\r\n  getOverallStoreHeatmapDates( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/zoneHeatmapAvailableDates_v1`, data)\r\n  }\r\n  getCardData( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/cardsAgeAnalysis_v1`, data)\r\n  }\r\n  getTopPerformingZonesData( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/topPerformingZones_v1`, data)\r\n  }\r\n  getTopPerformingStoresTableData( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/topPerformingStores_v1`, data)\r\n  }\r\n  getTopPerformingStoresTableExport( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/topPerformingStores_v1`, data, {responseType: 'arraybuffer'})\r\n  }\r\n  getSummaryTableData( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/zoneSummaryTable_v1`, data)\r\n  }\r\n  getSummaryTableExport( data:any):Observable<any>{\r\n    return this.http.post(`${this.zoneAnalysisUrl}/zoneSummaryTable_v1`, data, {responseType: 'arraybuffer'})\r\n  }\r\n\r\n  \r\n  \r\n\r\n\r\n  private handleError(error: HttpErrorResponse): Observable<never> {\r\n    return throwError(\r\n      () => new Error(\"Something bad happened; please try again later\")\r\n    );\r\n  }\r\n\r\n}\r\n"]}
|
|
@@ -1,11 +1,51 @@
|
|
|
1
1
|
import { Component } from '@angular/core';
|
|
2
|
+
import { Subject, debounceTime, takeUntil } from 'rxjs';
|
|
2
3
|
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "tango-app-ui-global";
|
|
5
|
+
import * as i2 from "@angular/common";
|
|
6
|
+
import * as i3 from "../overallcards/overallcards.component";
|
|
7
|
+
import * as i4 from "../top-performing-zones/top-performing-zones.component";
|
|
8
|
+
import * as i5 from "../zone-summary-table/zone-summary-table.component";
|
|
9
|
+
import * as i6 from "../zone-concentration/zone-concentration.component";
|
|
3
10
|
export class TangoAnalyseZoneComponent {
|
|
4
|
-
|
|
5
|
-
|
|
11
|
+
pageInfo;
|
|
12
|
+
gs;
|
|
13
|
+
storeId = 0;
|
|
14
|
+
headerData;
|
|
15
|
+
destroy$ = new Subject();
|
|
16
|
+
ngOnDestroy() {
|
|
17
|
+
this.destroy$.next(true);
|
|
18
|
+
this.destroy$.complete();
|
|
19
|
+
}
|
|
20
|
+
constructor(pageInfo, gs) {
|
|
21
|
+
this.pageInfo = pageInfo;
|
|
22
|
+
this.gs = gs;
|
|
23
|
+
}
|
|
24
|
+
ngOnInit() {
|
|
25
|
+
this.gs.dataRangeValue
|
|
26
|
+
.pipe(takeUntil(this.destroy$), debounceTime(300))
|
|
27
|
+
.subscribe((data) => {
|
|
28
|
+
this.headerData = data;
|
|
29
|
+
this.updateStoreId();
|
|
30
|
+
this.setPageData();
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
setPageData() {
|
|
34
|
+
this.pageInfo.setTitle("Zones");
|
|
35
|
+
this.pageInfo.setDescription("Zones");
|
|
36
|
+
this.pageInfo.setBreadcrumbs([
|
|
37
|
+
{ title: "Manage", path: "/manage", isActive: false, isSeparator: false },
|
|
38
|
+
{ title: "Manage", path: "/manage", isActive: false, isSeparator: true },
|
|
39
|
+
]);
|
|
40
|
+
}
|
|
41
|
+
updateStoreId() {
|
|
42
|
+
this.storeId = this.headerData.stores.filter((store) => store.checked).length;
|
|
43
|
+
}
|
|
44
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TangoAnalyseZoneComponent, deps: [{ token: i1.PageInfoService }, { token: i1.GlobalStateService }], target: i0.ɵɵFactoryTarget.Component });
|
|
45
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TangoAnalyseZoneComponent, selector: "lib-tango-analyse-zone", ngImport: i0, template: "<lib-zone-concentration></lib-zone-concentration>\r\n\r\n<!-- Conditionally display other components based on storeId value -->\r\n<div class=\"mt-4\">\r\n <lib-overallcards></lib-overallcards>\r\n</div>\r\n<div class=\"mt-2\" *ngIf=\"storeId > 1\">\r\n <lib-top-performing-zones></lib-top-performing-zones>\r\n</div>\r\n<div class=\"mt-4\" *ngIf=\"storeId > 1\">\r\n <lib-zone-summary-table></lib-zone-summary-table>\r\n</div>\r\n", styles: [""], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.OverallcardsComponent, selector: "lib-overallcards" }, { kind: "component", type: i4.TopPerformingZonesComponent, selector: "lib-top-performing-zones" }, { kind: "component", type: i5.ZoneSummaryTableComponent, selector: "lib-zone-summary-table" }, { kind: "component", type: i6.ZoneConcentrationComponent, selector: "lib-zone-concentration" }] });
|
|
6
46
|
}
|
|
7
47
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TangoAnalyseZoneComponent, decorators: [{
|
|
8
48
|
type: Component,
|
|
9
|
-
args: [{ selector: '
|
|
10
|
-
}] });
|
|
11
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
49
|
+
args: [{ selector: 'lib-tango-analyse-zone', template: "<lib-zone-concentration></lib-zone-concentration>\r\n\r\n<!-- Conditionally display other components based on storeId value -->\r\n<div class=\"mt-4\">\r\n <lib-overallcards></lib-overallcards>\r\n</div>\r\n<div class=\"mt-2\" *ngIf=\"storeId > 1\">\r\n <lib-top-performing-zones></lib-top-performing-zones>\r\n</div>\r\n<div class=\"mt-4\" *ngIf=\"storeId > 1\">\r\n <lib-zone-summary-table></lib-zone-summary-table>\r\n</div>\r\n" }]
|
|
50
|
+
}], ctorParameters: () => [{ type: i1.PageInfoService }, { type: i1.GlobalStateService }] });
|
|
51
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFuZ28tYW5hbHlzZS16b25lLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3RhbmdvLWFuYWx5c2Utem9uZS9zcmMvbGliL2NvbXBvbmVudHMvdGFuZ28tYW5hbHlzZS16b25lL3RhbmdvLWFuYWx5c2Utem9uZS5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90YW5nby1hbmFseXNlLXpvbmUvc3JjL2xpYi9jb21wb25lbnRzL3RhbmdvLWFuYWx5c2Utem9uZS90YW5nby1hbmFseXNlLXpvbmUuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUUxQyxPQUFPLEVBQWMsT0FBTyxFQUFFLFlBQVksRUFBd0IsU0FBUyxFQUFFLE1BQU0sTUFBTSxDQUFDOzs7Ozs7OztBQU0xRixNQUFNLE9BQU8seUJBQXlCO0lBVTFCO0lBQWlDO0lBVDNDLE9BQU8sR0FBVyxDQUFDLENBQUM7SUFDcEIsVUFBVSxDQUFNO0lBQ0MsUUFBUSxHQUFHLElBQUksT0FBTyxFQUFFLENBQUM7SUFFMUMsV0FBVztRQUNULElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ3pCLElBQUksQ0FBQyxRQUFRLENBQUMsUUFBUSxFQUFFLENBQUM7SUFDM0IsQ0FBQztJQUNELFlBQ1UsUUFBeUIsRUFBUSxFQUFzQjtRQUF2RCxhQUFRLEdBQVIsUUFBUSxDQUFpQjtRQUFRLE9BQUUsR0FBRixFQUFFLENBQW9CO0lBRWpFLENBQUM7SUFDRCxRQUFRO1FBQ04sSUFBSSxDQUFDLEVBQUUsQ0FBQyxjQUFjO2FBQ3JCLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxFQUFFLFlBQVksQ0FBQyxHQUFHLENBQUMsQ0FBQzthQUNqRCxTQUFTLENBQUMsQ0FBQyxJQUFTLEVBQUUsRUFBRTtZQUNyQixJQUFJLENBQUMsVUFBVSxHQUFHLElBQUksQ0FBQztZQUN2QixJQUFJLENBQUMsYUFBYSxFQUFFLENBQUE7WUFDcEIsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzNCLENBQUMsQ0FBQyxDQUFBO0lBQ0YsQ0FBQztJQUVDLFdBQVc7UUFDVCxJQUFJLENBQUMsUUFBUSxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNoQyxJQUFJLENBQUMsUUFBUSxDQUFDLGNBQWMsQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUN0QyxJQUFJLENBQUMsUUFBUSxDQUFDLGNBQWMsQ0FBQztZQUMzQixFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLFNBQVMsRUFBRSxRQUFRLEVBQUUsS0FBSyxFQUFFLFdBQVcsRUFBRSxLQUFLLEVBQUU7WUFDekUsRUFBRSxLQUFLLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxTQUFTLEVBQUUsUUFBUSxFQUFFLEtBQUssRUFBRSxXQUFXLEVBQUUsSUFBSSxFQUFFO1NBQ3pFLENBQUMsQ0FBQztJQUNMLENBQUM7SUFDRCxhQUFhO1FBQ1YsSUFBSSxDQUFDLE9BQU8sR0FBRyxJQUFJLENBQUMsVUFBVSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxLQUFVLEVBQUUsRUFBRSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQyxNQUFNLENBQUM7SUFDdEYsQ0FBQzt3R0FqQ1UseUJBQXlCOzRGQUF6Qix5QkFBeUIsOERDUnRDLG9iQVlBOzs0RkRKYSx5QkFBeUI7a0JBTHJDLFNBQVM7K0JBQ0Usd0JBQXdCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IEdsb2JhbFN0YXRlU2VydmljZSwgUGFnZUluZm9TZXJ2aWNlIH0gZnJvbSAndGFuZ28tYXBwLXVpLWdsb2JhbCc7XHJcbmltcG9ydCB7IE9ic2VydmFibGUsIFN1YmplY3QsIGRlYm91bmNlVGltZSwgZGlzdGluY3RVbnRpbENoYW5nZWQsIHRha2VVbnRpbCB9IGZyb20gJ3J4anMnO1xyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ2xpYi10YW5nby1hbmFseXNlLXpvbmUnLFxyXG4gIHRlbXBsYXRlVXJsOiAnLi90YW5nby1hbmFseXNlLXpvbmUuY29tcG9uZW50Lmh0bWwnLFxyXG4gIHN0eWxlVXJsOiAnLi90YW5nby1hbmFseXNlLXpvbmUuY29tcG9uZW50LnNjc3MnLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgVGFuZ29BbmFseXNlWm9uZUNvbXBvbmVudCB7XHJcbiAgc3RvcmVJZDogbnVtYmVyID0gMDtcclxuICBoZWFkZXJEYXRhOiBhbnk7XHJcbiAgcHJpdmF0ZSByZWFkb25seSBkZXN0cm95JCA9IG5ldyBTdWJqZWN0KCk7XHJcblxyXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xyXG4gICAgdGhpcy5kZXN0cm95JC5uZXh0KHRydWUpO1xyXG4gICAgdGhpcy5kZXN0cm95JC5jb21wbGV0ZSgpO1xyXG4gIH1cclxuICBjb25zdHJ1Y3RvcihcclxuICAgIHByaXZhdGUgcGFnZUluZm86IFBhZ2VJbmZvU2VydmljZSxwdWJsaWMgZ3M6IEdsb2JhbFN0YXRlU2VydmljZVxyXG4gICkge1xyXG4gIH1cclxuICBuZ09uSW5pdCgpOiB2b2lkIHtcclxuICAgIHRoaXMuZ3MuZGF0YVJhbmdlVmFsdWVcclxuICAgIC5waXBlKHRha2VVbnRpbCh0aGlzLmRlc3Ryb3kkKSwgZGVib3VuY2VUaW1lKDMwMCkpXHJcbiAgICAuc3Vic2NyaWJlKChkYXRhOiBhbnkpID0+IHtcclxuICAgICAgICB0aGlzLmhlYWRlckRhdGEgPSBkYXRhO1xyXG4gICAgICAgIHRoaXMudXBkYXRlU3RvcmVJZCgpXHJcbiAgICAgICAgdGhpcy5zZXRQYWdlRGF0YSgpO1xyXG59KVxyXG59XHJcblxyXG4gIHNldFBhZ2VEYXRhKCkge1xyXG4gICAgdGhpcy5wYWdlSW5mby5zZXRUaXRsZShcIlpvbmVzXCIpO1xyXG4gICAgdGhpcy5wYWdlSW5mby5zZXREZXNjcmlwdGlvbihcIlpvbmVzXCIpO1xyXG4gICAgdGhpcy5wYWdlSW5mby5zZXRCcmVhZGNydW1icyhbXHJcbiAgICAgIHsgdGl0bGU6IFwiTWFuYWdlXCIsIHBhdGg6IFwiL21hbmFnZVwiLCBpc0FjdGl2ZTogZmFsc2UsIGlzU2VwYXJhdG9yOiBmYWxzZSB9LFxyXG4gICAgICB7IHRpdGxlOiBcIk1hbmFnZVwiLCBwYXRoOiBcIi9tYW5hZ2VcIiwgaXNBY3RpdmU6IGZhbHNlLCBpc1NlcGFyYXRvcjogdHJ1ZSB9LFxyXG4gICAgXSk7XHJcbiAgfVxyXG4gIHVwZGF0ZVN0b3JlSWQoKTogdm9pZCB7XHJcbiAgICAgdGhpcy5zdG9yZUlkID0gdGhpcy5oZWFkZXJEYXRhLnN0b3Jlcy5maWx0ZXIoKHN0b3JlOiBhbnkpID0+IHN0b3JlLmNoZWNrZWQpLmxlbmd0aDtcclxuICB9XHJcbn1cclxuIiwiPGxpYi16b25lLWNvbmNlbnRyYXRpb24+PC9saWItem9uZS1jb25jZW50cmF0aW9uPlxyXG5cclxuPCEtLSBDb25kaXRpb25hbGx5IGRpc3BsYXkgb3RoZXIgY29tcG9uZW50cyBiYXNlZCBvbiBzdG9yZUlkIHZhbHVlIC0tPlxyXG48ZGl2IGNsYXNzPVwibXQtNFwiPlxyXG4gIDxsaWItb3ZlcmFsbGNhcmRzPjwvbGliLW92ZXJhbGxjYXJkcz5cclxuPC9kaXY+XHJcbjxkaXYgY2xhc3M9XCJtdC0yXCIgKm5nSWY9XCJzdG9yZUlkID4gMVwiPlxyXG4gIDxsaWItdG9wLXBlcmZvcm1pbmctem9uZXM+PC9saWItdG9wLXBlcmZvcm1pbmctem9uZXM+XHJcbjwvZGl2PlxyXG48ZGl2IGNsYXNzPVwibXQtNFwiICpuZ0lmPVwic3RvcmVJZCA+IDFcIj5cclxuICA8bGliLXpvbmUtc3VtbWFyeS10YWJsZT48L2xpYi16b25lLXN1bW1hcnktdGFibGU+XHJcbjwvZGl2PlxyXG4iXX0=
|