tango-app-ui-store-builder 1.0.28 → 1.0.30
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/collection-update-ai/collection-update-ai.component.mjs +1486 -0
- package/esm2022/lib/components/manage-plano/popups/create-planogram/create-planogram.component.mjs +6 -9
- package/esm2022/lib/components/manage-store-plano/manage-store-plano.component.mjs +203 -76
- package/esm2022/lib/services/store-builder.service.mjs +10 -1
- package/esm2022/lib/tango-store-builder-routing.module.mjs +6 -1
- package/esm2022/lib/tango-store-builder.module.mjs +1 -4
- package/esm2022/lib/tango-store-plano.module.mjs +18 -20
- package/fesm2022/tango-app-ui-store-builder.mjs +2167 -561
- package/fesm2022/tango-app-ui-store-builder.mjs.map +1 -1
- package/lib/components/collection-update-ai/collection-update-ai.component.d.ts +82 -0
- package/lib/components/manage-plano/popups/create-planogram/create-planogram.component.d.ts +1 -0
- package/lib/components/manage-plano/rollout-table/rollout-table.component.d.ts +2 -2
- package/lib/components/manage-plano/verification-table/verification-table.component.d.ts +4 -4
- package/lib/components/manage-store-plano/manage-store-plano.component.d.ts +11 -2
- package/lib/services/store-builder.service.d.ts +3 -0
- package/package.json +1 -1
|
@@ -0,0 +1,1486 @@
|
|
|
1
|
+
import { Component, ViewChild } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { ReactiveFormsModule, Validators } from '@angular/forms';
|
|
4
|
+
import { lastValueFrom, Subject, takeUntil } from 'rxjs';
|
|
5
|
+
import * as fabric from "fabric";
|
|
6
|
+
import { TitleCasePipe } from "@angular/common";
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
import * as i1 from "@angular/forms";
|
|
9
|
+
import * as i2 from "../../services/store-builder.service";
|
|
10
|
+
import * as i3 from "tango-app-ui-shared";
|
|
11
|
+
import * as i4 from "@angular/router";
|
|
12
|
+
import * as i5 from "@angular/common";
|
|
13
|
+
export class CollectionUpdateAiComponent {
|
|
14
|
+
fb;
|
|
15
|
+
apiService;
|
|
16
|
+
toastService;
|
|
17
|
+
route;
|
|
18
|
+
titleCase;
|
|
19
|
+
form;
|
|
20
|
+
loading = false;
|
|
21
|
+
beforeCanvasRef;
|
|
22
|
+
beforeCanvas;
|
|
23
|
+
beforeContainerRef;
|
|
24
|
+
afterCanvasRef;
|
|
25
|
+
afterCanvas;
|
|
26
|
+
afterContainerRef;
|
|
27
|
+
destroy$ = new Subject();
|
|
28
|
+
planoData;
|
|
29
|
+
floorData;
|
|
30
|
+
wallThickness = 30;
|
|
31
|
+
verticalMargin = (1.25 / 100) * this.toPixels(1524, "mm");
|
|
32
|
+
cdnUrl;
|
|
33
|
+
activeHighlight = null;
|
|
34
|
+
cancelHighlightBlink = false;
|
|
35
|
+
storeChips = [];
|
|
36
|
+
selectedStore;
|
|
37
|
+
selectedFixture;
|
|
38
|
+
constructor(fb, apiService, toastService, route, titleCase) {
|
|
39
|
+
this.fb = fb;
|
|
40
|
+
this.apiService = apiService;
|
|
41
|
+
this.toastService = toastService;
|
|
42
|
+
this.route = route;
|
|
43
|
+
this.titleCase = titleCase;
|
|
44
|
+
this.form = this.fb.group({
|
|
45
|
+
stores: ['', Validators.required],
|
|
46
|
+
updateType: ['vm', Validators.required],
|
|
47
|
+
selectionCriteria: ['', Validators.required],
|
|
48
|
+
updationCriteria: ['', Validators.required],
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
ngAfterViewInit() {
|
|
52
|
+
this.route.queryParams
|
|
53
|
+
.pipe(takeUntil(this.destroy$))
|
|
54
|
+
.subscribe((params) => {
|
|
55
|
+
if (params?.planoId) {
|
|
56
|
+
this.getStoreFixtures(params?.planoId);
|
|
57
|
+
if (params?.floorId) {
|
|
58
|
+
this.floorData = { _id: params?.floorId };
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
});
|
|
62
|
+
this.beforeCanvas = new fabric.Canvas(this.beforeCanvasRef?.nativeElement, {
|
|
63
|
+
selectionFullyContained: true,
|
|
64
|
+
perPixelTargetFind: true,
|
|
65
|
+
targetFindTolerance: 5,
|
|
66
|
+
});
|
|
67
|
+
this.afterCanvas = new fabric.Canvas(this.afterCanvasRef?.nativeElement, {
|
|
68
|
+
selectionFullyContained: true,
|
|
69
|
+
perPixelTargetFind: true,
|
|
70
|
+
targetFindTolerance: 5,
|
|
71
|
+
});
|
|
72
|
+
this.setupZoomHandlers(this.beforeCanvas);
|
|
73
|
+
this.setupZoomHandlers(this.afterCanvas);
|
|
74
|
+
}
|
|
75
|
+
setupZoomHandlers(canvas) {
|
|
76
|
+
canvas.on("mouse:wheel", (opt) => {
|
|
77
|
+
const e = opt.e;
|
|
78
|
+
if (e.ctrlKey || e.metaKey) {
|
|
79
|
+
const delta = e.deltaY;
|
|
80
|
+
let zoom = canvas.getZoom();
|
|
81
|
+
zoom *= 0.995 ** delta;
|
|
82
|
+
zoom = Math.max(0.2, Math.min(zoom, 3));
|
|
83
|
+
canvas.zoomToPoint({
|
|
84
|
+
x: e.offsetX,
|
|
85
|
+
y: e.offsetY,
|
|
86
|
+
}, zoom);
|
|
87
|
+
e.preventDefault();
|
|
88
|
+
e.stopPropagation();
|
|
89
|
+
}
|
|
90
|
+
});
|
|
91
|
+
let isDragging = false;
|
|
92
|
+
let lastPosX = 0;
|
|
93
|
+
let lastPosY = 0;
|
|
94
|
+
canvas.on("mouse:down", (opt) => {
|
|
95
|
+
if (!opt.target) {
|
|
96
|
+
isDragging = true;
|
|
97
|
+
canvas.selection = false;
|
|
98
|
+
const e = opt.e;
|
|
99
|
+
lastPosX = e.clientX;
|
|
100
|
+
lastPosY = e.clientY;
|
|
101
|
+
}
|
|
102
|
+
});
|
|
103
|
+
canvas.on("mouse:move", (opt) => {
|
|
104
|
+
if (isDragging && canvas.viewportTransform) {
|
|
105
|
+
const e = opt.e;
|
|
106
|
+
const vpt = canvas.viewportTransform;
|
|
107
|
+
vpt[4] += e.clientX - lastPosX;
|
|
108
|
+
vpt[5] += e.clientY - lastPosY;
|
|
109
|
+
canvas.requestRenderAll();
|
|
110
|
+
lastPosX = e.clientX;
|
|
111
|
+
lastPosY = e.clientY;
|
|
112
|
+
}
|
|
113
|
+
});
|
|
114
|
+
canvas.on("mouse:up", () => {
|
|
115
|
+
isDragging = false;
|
|
116
|
+
canvas.selection = true;
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
updateAiResponseFixture(fixture) {
|
|
120
|
+
let matchFound = false;
|
|
121
|
+
for (let i = 0; i < this.floorData.layoutPolygon.length; i++) {
|
|
122
|
+
const element = this.floorData.layoutPolygon[i];
|
|
123
|
+
if (element.elementType === 'wall' && Array.isArray(element.fixtures)) {
|
|
124
|
+
for (let j = 0; j < element.fixtures.length; j++) {
|
|
125
|
+
const fx = element.fixtures[j];
|
|
126
|
+
if (fx._id === fixture._id) {
|
|
127
|
+
this.floorData.layoutPolygon[i].fixtures[j] = {
|
|
128
|
+
...fixture
|
|
129
|
+
};
|
|
130
|
+
matchFound = true;
|
|
131
|
+
this.selectedFixture = fixture;
|
|
132
|
+
break;
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
if (matchFound)
|
|
138
|
+
return;
|
|
139
|
+
for (let i = 0; i < this.floorData?.centerFixture.length; i++) {
|
|
140
|
+
const fx = this.floorData?.centerFixture[i];
|
|
141
|
+
if (fx._id === fixture._id) {
|
|
142
|
+
this.floorData.centerFixture[i] = {
|
|
143
|
+
...fixture
|
|
144
|
+
};
|
|
145
|
+
this.selectedFixture = fixture;
|
|
146
|
+
break;
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
submitPrompt() {
|
|
151
|
+
if (this.form.invalid) {
|
|
152
|
+
this.form.markAllAsTouched();
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
this.loading = true;
|
|
156
|
+
const payload = {
|
|
157
|
+
...this.form.value,
|
|
158
|
+
stores: [this.selectedStore]
|
|
159
|
+
};
|
|
160
|
+
this.apiService.findFixtureAi(payload).subscribe({
|
|
161
|
+
next: (res) => {
|
|
162
|
+
this.loading = false;
|
|
163
|
+
if (res.code === 200) {
|
|
164
|
+
this.updateAiResponseFixture(res.data);
|
|
165
|
+
this.renderFloor(this.afterCanvas);
|
|
166
|
+
setTimeout(() => {
|
|
167
|
+
const actionFixture = this.afterCanvas
|
|
168
|
+
.getObjects()
|
|
169
|
+
.find((obj) => {
|
|
170
|
+
return obj.objType === "fixture" && obj.fixtureId === res.data._id;
|
|
171
|
+
});
|
|
172
|
+
if (!actionFixture)
|
|
173
|
+
return;
|
|
174
|
+
this.highlightFixture(this.afterCanvas, actionFixture, true);
|
|
175
|
+
}, 0);
|
|
176
|
+
}
|
|
177
|
+
},
|
|
178
|
+
error: () => {
|
|
179
|
+
this.loading = false;
|
|
180
|
+
},
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
async getStoreFixtures(storeId) {
|
|
184
|
+
this.loading = true;
|
|
185
|
+
this.apiService
|
|
186
|
+
.getStoreFixtures({ id: [storeId] })
|
|
187
|
+
.pipe(takeUntil(this.destroy$))
|
|
188
|
+
.subscribe({
|
|
189
|
+
next: async (res) => {
|
|
190
|
+
if (!res?.data?.[0]?.floors?.length) {
|
|
191
|
+
return;
|
|
192
|
+
}
|
|
193
|
+
this.planoData = structuredClone(res.data[0]);
|
|
194
|
+
this.floorData = structuredClone(res.data[0].floors[0]);
|
|
195
|
+
this.renderFloor(this.beforeCanvas);
|
|
196
|
+
this.renderFloor(this.afterCanvas);
|
|
197
|
+
this.loading = false;
|
|
198
|
+
setTimeout(() => {
|
|
199
|
+
this.resizeCanvas(this.beforeCanvas, this.beforeContainerRef);
|
|
200
|
+
this.fitCanvasToLayoutAnimated(this.beforeCanvas);
|
|
201
|
+
this.resizeCanvas(this.afterCanvas, this.afterContainerRef);
|
|
202
|
+
this.fitCanvasToLayoutAnimated(this.afterCanvas);
|
|
203
|
+
}, 300);
|
|
204
|
+
},
|
|
205
|
+
error: (err) => {
|
|
206
|
+
this.loading = false;
|
|
207
|
+
this.toastService.getErrorToast("Failed to fetch plano");
|
|
208
|
+
},
|
|
209
|
+
});
|
|
210
|
+
}
|
|
211
|
+
ngOnDestroy() {
|
|
212
|
+
this.destroy$.next(true);
|
|
213
|
+
this.destroy$.complete();
|
|
214
|
+
}
|
|
215
|
+
resizeCanvas(canvas, conteiner) {
|
|
216
|
+
const container = conteiner.nativeElement;
|
|
217
|
+
canvas.setDimensions({
|
|
218
|
+
width: container.clientWidth - 50,
|
|
219
|
+
height: container.clientHeight - 40,
|
|
220
|
+
});
|
|
221
|
+
canvas.requestRenderAll();
|
|
222
|
+
}
|
|
223
|
+
fitCanvasToLayoutAnimated(canvas) {
|
|
224
|
+
if (!canvas)
|
|
225
|
+
return;
|
|
226
|
+
const bounds = this.getFullLayoutBounds(canvas);
|
|
227
|
+
if (!bounds.width || !bounds.height)
|
|
228
|
+
return;
|
|
229
|
+
const canvasW = canvas.getWidth();
|
|
230
|
+
const canvasH = canvas.getHeight();
|
|
231
|
+
// Final zoom to fit (with padding)
|
|
232
|
+
const scaleX = canvasW / bounds.width;
|
|
233
|
+
const scaleY = canvasH / bounds.height;
|
|
234
|
+
const targetZoom = Math.min(scaleX, scaleY) * 0.9; // padding 10%
|
|
235
|
+
// Desired viewport offset to center layout
|
|
236
|
+
const targetVpt = [
|
|
237
|
+
...(canvas.viewportTransform || [1, 0, 0, 1, 0, 0]),
|
|
238
|
+
];
|
|
239
|
+
targetVpt[0] = targetZoom; // scaleX
|
|
240
|
+
targetVpt[3] = targetZoom; // scaleY
|
|
241
|
+
targetVpt[4] = canvasW / 2 - (bounds.left + bounds.width / 2) * targetZoom; // center X
|
|
242
|
+
targetVpt[5] = canvasH / 2 - (bounds.top + bounds.height / 2) * targetZoom; // center Y
|
|
243
|
+
// Current viewport (starting point of animation)
|
|
244
|
+
const startVpt = [...canvas.viewportTransform];
|
|
245
|
+
fabric.util.animate({
|
|
246
|
+
startValue: 0,
|
|
247
|
+
endValue: 1,
|
|
248
|
+
duration: 100,
|
|
249
|
+
easing: fabric.util.ease.easeOutCubic,
|
|
250
|
+
onChange: (value) => {
|
|
251
|
+
const vpt = canvas.viewportTransform;
|
|
252
|
+
// Interpolate matrix values
|
|
253
|
+
vpt[0] = startVpt[0] + (targetVpt[0] - startVpt[0]) * value;
|
|
254
|
+
vpt[3] = startVpt[3] + (targetVpt[3] - startVpt[3]) * value;
|
|
255
|
+
vpt[4] = startVpt[4] + (targetVpt[4] - startVpt[4]) * value;
|
|
256
|
+
vpt[5] = startVpt[5] + (targetVpt[5] - startVpt[5]) * value;
|
|
257
|
+
canvas.requestRenderAll();
|
|
258
|
+
},
|
|
259
|
+
});
|
|
260
|
+
}
|
|
261
|
+
getFullLayoutBounds(canvas) {
|
|
262
|
+
const objects = canvas.getObjects();
|
|
263
|
+
if (!objects.length)
|
|
264
|
+
return { left: 0, top: 0, width: 0, height: 0 };
|
|
265
|
+
let minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity;
|
|
266
|
+
for (const obj of objects) {
|
|
267
|
+
const rect = obj.getBoundingRect();
|
|
268
|
+
minX = Math.min(minX, rect.left);
|
|
269
|
+
minY = Math.min(minY, rect.top);
|
|
270
|
+
maxX = Math.max(maxX, rect.left + rect.width);
|
|
271
|
+
maxY = Math.max(maxY, rect.top + rect.height);
|
|
272
|
+
}
|
|
273
|
+
return {
|
|
274
|
+
left: minX,
|
|
275
|
+
top: minY,
|
|
276
|
+
width: maxX - minX,
|
|
277
|
+
height: maxY - minY,
|
|
278
|
+
};
|
|
279
|
+
}
|
|
280
|
+
renderFloor(canvas) {
|
|
281
|
+
this.drawInitialContent(canvas);
|
|
282
|
+
}
|
|
283
|
+
async drawInitialContent(canvas) {
|
|
284
|
+
canvas.clear();
|
|
285
|
+
await this.drawLayoutElements(canvas);
|
|
286
|
+
// this.drawBg();
|
|
287
|
+
}
|
|
288
|
+
async drawLayoutElements(canvas) {
|
|
289
|
+
for (let index = 0; index < this.floorData.layoutPolygon.length; index++) {
|
|
290
|
+
const element = this.floorData.layoutPolygon[index];
|
|
291
|
+
let wall;
|
|
292
|
+
element.relativePosition ??= {
|
|
293
|
+
x: 0,
|
|
294
|
+
y: 0,
|
|
295
|
+
unit: "ft",
|
|
296
|
+
};
|
|
297
|
+
const x = this.toPixels(element.relativePosition.x, element.unit);
|
|
298
|
+
const y = this.toPixels(element.relativePosition.y, element.unit);
|
|
299
|
+
if (element.elementType === "wall") {
|
|
300
|
+
const height = this.wallThickness;
|
|
301
|
+
const width = this.toPixels(element.distance, element.unit);
|
|
302
|
+
wall = await this.drawWall(canvas, x, y, height, width, element);
|
|
303
|
+
}
|
|
304
|
+
else if (element.elementType === "entrance") {
|
|
305
|
+
const arrowWidth = 80;
|
|
306
|
+
const arrowHeight = 75;
|
|
307
|
+
const arrowBase = 40;
|
|
308
|
+
this.drawEntrance(canvas, x, y, arrowWidth, arrowHeight, arrowBase, element);
|
|
309
|
+
}
|
|
310
|
+
if (element.elementType === "wall") {
|
|
311
|
+
const fixtureGroups = [];
|
|
312
|
+
for (let j = 0; j < element?.fixtures?.length; j++) {
|
|
313
|
+
const fixture = element.fixtures[j];
|
|
314
|
+
const x = this.toPixels(fixture.relativePosition.x, fixture.relativePosition.unit);
|
|
315
|
+
const y = this.toPixels(fixture.relativePosition.y, fixture.relativePosition.unit);
|
|
316
|
+
const fixtureHeight = this.toPixels(fixture.fixtureStaticLength.value, fixture.fixtureStaticLength.unit);
|
|
317
|
+
const fixtureWidth = this.toPixels(fixture.fixtureStaticWidth.value, fixture.fixtureStaticWidth.unit);
|
|
318
|
+
const fixtureObj = await this.drawFixture(canvas, x, y, fixtureHeight, fixtureWidth, fixture);
|
|
319
|
+
fixtureGroups.push(fixtureObj);
|
|
320
|
+
}
|
|
321
|
+
if (element.isLocked && wall) {
|
|
322
|
+
canvas.remove(wall);
|
|
323
|
+
fixtureGroups.forEach((f) => canvas.remove(f));
|
|
324
|
+
const combinedGroup = new fabric.Group([wall, ...fixtureGroups], {
|
|
325
|
+
hasControls: true,
|
|
326
|
+
selectable: true,
|
|
327
|
+
subTargetCheck: true,
|
|
328
|
+
});
|
|
329
|
+
combinedGroup.set({ isLocked: true });
|
|
330
|
+
canvas.add(combinedGroup);
|
|
331
|
+
}
|
|
332
|
+
for (let j = 0; j < element?.otherElements?.length; j++) {
|
|
333
|
+
const otherElement = element.otherElements[j];
|
|
334
|
+
const x = this.toPixels(otherElement.relativePosition.x, "ft");
|
|
335
|
+
const y = this.toPixels(otherElement.relativePosition.y, "ft");
|
|
336
|
+
const a = otherElement.relativePosition.angle;
|
|
337
|
+
const h = this.toPixels(otherElement.fixtureStaticLength.value, otherElement.fixtureStaticLength.unit);
|
|
338
|
+
const w = this.toPixels(otherElement.fixtureStaticWidth.value, otherElement.fixtureStaticWidth.unit);
|
|
339
|
+
const fixtureObj = this.drawOtherElementsWithLabel(canvas, x, y, a, w, h, otherElement);
|
|
340
|
+
// elementsGroup.push(fixtureObj as fabric.Group);
|
|
341
|
+
}
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
for (let i = 0; i < this.floorData?.centerFixture?.length; i++) {
|
|
345
|
+
const fixture = this.floorData.centerFixture[i];
|
|
346
|
+
const x = this.toPixels(fixture.relativePosition.x, fixture.relativePosition.unit);
|
|
347
|
+
const y = this.toPixels(fixture.relativePosition.y, fixture.relativePosition.unit);
|
|
348
|
+
const fixtureHeight = this.toPixels(fixture.fixtureStaticLength.value, fixture.fixtureStaticLength.unit);
|
|
349
|
+
const fixtureWidth = this.toPixels(fixture.fixtureStaticWidth.value, fixture.fixtureStaticWidth.unit);
|
|
350
|
+
await this.drawFixture(canvas, x, y, fixtureHeight, fixtureWidth, fixture);
|
|
351
|
+
}
|
|
352
|
+
for (let j = 0; j < this.floorData?.otherElements?.length; j++) {
|
|
353
|
+
const otherElement = this.floorData.otherElements[j];
|
|
354
|
+
const x = this.toPixels(otherElement.relativePosition.x, "ft");
|
|
355
|
+
const y = this.toPixels(otherElement.relativePosition.y, "ft");
|
|
356
|
+
const a = otherElement.relativePosition.angle;
|
|
357
|
+
const h = this.toPixels(otherElement.fixtureStaticLength.value, otherElement.fixtureStaticLength.unit);
|
|
358
|
+
const w = this.toPixels(otherElement.fixtureStaticWidth.value, otherElement.fixtureStaticWidth.unit);
|
|
359
|
+
const fixtureObj = this.drawOtherElementsWithLabel(canvas, x, y, a, w, h, otherElement);
|
|
360
|
+
}
|
|
361
|
+
return true;
|
|
362
|
+
}
|
|
363
|
+
toPixels(value, unit, dpi = 2.8) {
|
|
364
|
+
let pixels;
|
|
365
|
+
if (unit === "mm") {
|
|
366
|
+
pixels = (value / 25.4) * dpi;
|
|
367
|
+
}
|
|
368
|
+
else if (unit === "ft") {
|
|
369
|
+
pixels = value * 12 * dpi;
|
|
370
|
+
}
|
|
371
|
+
else {
|
|
372
|
+
throw new Error("Unsupported unit. Use 'mm' or 'ft'.");
|
|
373
|
+
}
|
|
374
|
+
return Number(pixels.toFixed(4));
|
|
375
|
+
}
|
|
376
|
+
async drawWall(canvas, x, y, height, width, elementData) {
|
|
377
|
+
const radius = 5;
|
|
378
|
+
const fillColor = "#F6FCFF";
|
|
379
|
+
const strokeColor = "#D0D5DD";
|
|
380
|
+
const labelColor = "#009BF3";
|
|
381
|
+
let text = `${this.titleCase.transform(elementData.elementType)} ${elementData.elementNumber}`;
|
|
382
|
+
const wall = new fabric.Rect({
|
|
383
|
+
left: 0,
|
|
384
|
+
top: 0,
|
|
385
|
+
width,
|
|
386
|
+
height,
|
|
387
|
+
rx: radius,
|
|
388
|
+
ry: radius,
|
|
389
|
+
fill: fillColor,
|
|
390
|
+
stroke: strokeColor,
|
|
391
|
+
strokeWidth: 0.27,
|
|
392
|
+
selectable: false,
|
|
393
|
+
strokeUniform: true,
|
|
394
|
+
});
|
|
395
|
+
const centerX = width / 2;
|
|
396
|
+
const centerY = height / 2;
|
|
397
|
+
const isVertical = height > width;
|
|
398
|
+
const label = new fabric.FabricText(text, {
|
|
399
|
+
left: centerX,
|
|
400
|
+
top: centerY,
|
|
401
|
+
fontSize: 10,
|
|
402
|
+
fontFamily: "Inter",
|
|
403
|
+
fontWeight: "500",
|
|
404
|
+
fill: labelColor,
|
|
405
|
+
originX: "center",
|
|
406
|
+
originY: "center",
|
|
407
|
+
angle: isVertical ? -90 : 0,
|
|
408
|
+
selectable: false,
|
|
409
|
+
});
|
|
410
|
+
const arrowHead = 5;
|
|
411
|
+
const elements = [wall, label];
|
|
412
|
+
if (isVertical) {
|
|
413
|
+
const textWidth = label.width || 0;
|
|
414
|
+
const topArrowY = 5;
|
|
415
|
+
const topArrowEndY = centerY - textWidth / 2 - 5;
|
|
416
|
+
if (topArrowEndY > topArrowY + arrowHead) {
|
|
417
|
+
elements.push(new fabric.Line([centerX, topArrowY, centerX, topArrowEndY], {
|
|
418
|
+
stroke: labelColor,
|
|
419
|
+
strokeWidth: 1,
|
|
420
|
+
selectable: false,
|
|
421
|
+
}), new fabric.Triangle({
|
|
422
|
+
left: centerX,
|
|
423
|
+
top: topArrowY,
|
|
424
|
+
width: arrowHead,
|
|
425
|
+
height: arrowHead,
|
|
426
|
+
angle: 0,
|
|
427
|
+
fill: labelColor,
|
|
428
|
+
originX: "center",
|
|
429
|
+
originY: "bottom",
|
|
430
|
+
selectable: false,
|
|
431
|
+
}));
|
|
432
|
+
}
|
|
433
|
+
const bottomArrowStartY = centerY + textWidth / 2 + 5;
|
|
434
|
+
const bottomArrowEndY = height - 5;
|
|
435
|
+
if (bottomArrowEndY > bottomArrowStartY + arrowHead) {
|
|
436
|
+
elements.push(new fabric.Line([centerX, bottomArrowStartY, centerX, bottomArrowEndY], {
|
|
437
|
+
stroke: labelColor,
|
|
438
|
+
strokeWidth: 1,
|
|
439
|
+
selectable: false,
|
|
440
|
+
}), new fabric.Triangle({
|
|
441
|
+
left: centerX,
|
|
442
|
+
top: bottomArrowEndY,
|
|
443
|
+
width: arrowHead,
|
|
444
|
+
height: arrowHead,
|
|
445
|
+
angle: 180,
|
|
446
|
+
fill: labelColor,
|
|
447
|
+
originX: "center",
|
|
448
|
+
originY: "top",
|
|
449
|
+
selectable: false,
|
|
450
|
+
}));
|
|
451
|
+
}
|
|
452
|
+
}
|
|
453
|
+
else {
|
|
454
|
+
const textWidth = label.width || 0;
|
|
455
|
+
const leftArrowStartX = 5;
|
|
456
|
+
const leftArrowEndX = centerX - textWidth / 2 - 5;
|
|
457
|
+
if (leftArrowEndX > leftArrowStartX + arrowHead) {
|
|
458
|
+
elements.push(new fabric.Line([leftArrowStartX, centerY, leftArrowEndX, centerY], {
|
|
459
|
+
stroke: labelColor,
|
|
460
|
+
strokeWidth: 1,
|
|
461
|
+
selectable: false,
|
|
462
|
+
}), new fabric.Triangle({
|
|
463
|
+
left: leftArrowStartX,
|
|
464
|
+
top: centerY,
|
|
465
|
+
width: arrowHead,
|
|
466
|
+
height: arrowHead,
|
|
467
|
+
angle: -90,
|
|
468
|
+
fill: labelColor,
|
|
469
|
+
originX: "center",
|
|
470
|
+
originY: "center",
|
|
471
|
+
selectable: false,
|
|
472
|
+
}));
|
|
473
|
+
}
|
|
474
|
+
const rightArrowStartX = centerX + textWidth / 2 + 5;
|
|
475
|
+
const rightArrowEndX = width - 5;
|
|
476
|
+
if (rightArrowEndX > rightArrowStartX + arrowHead) {
|
|
477
|
+
elements.push(new fabric.Line([rightArrowStartX, centerY, rightArrowEndX, centerY], {
|
|
478
|
+
stroke: labelColor,
|
|
479
|
+
strokeWidth: 1,
|
|
480
|
+
selectable: false,
|
|
481
|
+
}), new fabric.Triangle({
|
|
482
|
+
left: rightArrowEndX,
|
|
483
|
+
top: centerY,
|
|
484
|
+
width: arrowHead,
|
|
485
|
+
height: arrowHead,
|
|
486
|
+
angle: 90,
|
|
487
|
+
fill: labelColor,
|
|
488
|
+
originX: "center",
|
|
489
|
+
originY: "center",
|
|
490
|
+
selectable: false,
|
|
491
|
+
}));
|
|
492
|
+
}
|
|
493
|
+
}
|
|
494
|
+
const group = new fabric.Group(elements, {
|
|
495
|
+
left: x,
|
|
496
|
+
top: y,
|
|
497
|
+
hasControls: true,
|
|
498
|
+
lockScalingX: isVertical,
|
|
499
|
+
lockScalingY: !isVertical,
|
|
500
|
+
// lockRotation: true,
|
|
501
|
+
evented: true,
|
|
502
|
+
hoverCursor: "pointer",
|
|
503
|
+
angle: elementData.angle,
|
|
504
|
+
subTargetCheck: true,
|
|
505
|
+
});
|
|
506
|
+
group.objType = elementData.elementType;
|
|
507
|
+
group.data = elementData;
|
|
508
|
+
canvas.add(group);
|
|
509
|
+
// If newly added, focus it once
|
|
510
|
+
if (elementData?.newAdded) {
|
|
511
|
+
try {
|
|
512
|
+
canvas.setActiveObject(group);
|
|
513
|
+
canvas.requestRenderAll();
|
|
514
|
+
}
|
|
515
|
+
catch { }
|
|
516
|
+
delete elementData.newAdded;
|
|
517
|
+
}
|
|
518
|
+
return group;
|
|
519
|
+
}
|
|
520
|
+
drawEntrance(canvas, x, y, width, height, arrowWidth, elementData) {
|
|
521
|
+
const bodyWidth = width - arrowWidth;
|
|
522
|
+
const pathData = [
|
|
523
|
+
`M 0 0`,
|
|
524
|
+
`L ${bodyWidth} 0`,
|
|
525
|
+
`L ${bodyWidth} ${-height / 2}`,
|
|
526
|
+
`L ${width} ${height / 2}`,
|
|
527
|
+
`L ${bodyWidth} ${height + height / 2}`,
|
|
528
|
+
`L ${bodyWidth} ${height}`,
|
|
529
|
+
`L 0 ${height}`,
|
|
530
|
+
`Z`,
|
|
531
|
+
].join(" ");
|
|
532
|
+
const arrow = new fabric.Path(pathData, {
|
|
533
|
+
fill: "#03A9F4",
|
|
534
|
+
selectable: false,
|
|
535
|
+
evented: false,
|
|
536
|
+
});
|
|
537
|
+
const maxTextHeight = height - 6;
|
|
538
|
+
const text = new fabric.Text("Entrance", {
|
|
539
|
+
fontFamily: "Arial",
|
|
540
|
+
fontWeight: "bold",
|
|
541
|
+
fill: "white",
|
|
542
|
+
angle: -90,
|
|
543
|
+
originX: "center",
|
|
544
|
+
originY: "center",
|
|
545
|
+
left: 15,
|
|
546
|
+
top: height / 2,
|
|
547
|
+
fontSize: 20,
|
|
548
|
+
selectable: false,
|
|
549
|
+
evented: false,
|
|
550
|
+
});
|
|
551
|
+
const textBounds = text.getBoundingRect();
|
|
552
|
+
if (textBounds.height > maxTextHeight) {
|
|
553
|
+
const scale = maxTextHeight / textBounds.height;
|
|
554
|
+
text.set("fontSize", text.fontSize * scale);
|
|
555
|
+
}
|
|
556
|
+
const group = new fabric.Group([arrow, text], {
|
|
557
|
+
left: x,
|
|
558
|
+
top: y,
|
|
559
|
+
lockScalingX: true,
|
|
560
|
+
lockScalingY: true,
|
|
561
|
+
lockScalingFlip: true,
|
|
562
|
+
angle: elementData.angle,
|
|
563
|
+
});
|
|
564
|
+
group.objType = elementData.elementType;
|
|
565
|
+
group.data = elementData;
|
|
566
|
+
canvas.add(group);
|
|
567
|
+
}
|
|
568
|
+
async drawFixture(canvas, x, y, height, width, fixtureData) {
|
|
569
|
+
let fixtureColor;
|
|
570
|
+
// if (["fixture", "fixture-redo"].includes(this.publishingState)) {
|
|
571
|
+
// fixtureColor = {
|
|
572
|
+
// primary:
|
|
573
|
+
// fixtureData.status == "" && fixtureData.taskStatus == "submit"
|
|
574
|
+
// ? "#a5a6a8"
|
|
575
|
+
// : fixtureData.status == "agree"
|
|
576
|
+
// ? "#6CE9A6"
|
|
577
|
+
// : "#51C1FF",
|
|
578
|
+
// fill:
|
|
579
|
+
// fixtureData.status == "" && fixtureData.taskStatus == "submit"
|
|
580
|
+
// ? "#f3f4f8"
|
|
581
|
+
// : fixtureData.status == "agree"
|
|
582
|
+
// ? "#6CE9A6"
|
|
583
|
+
// : "#EAF8FF",
|
|
584
|
+
// vmFill:
|
|
585
|
+
// fixtureData.status == "" && fixtureData.taskStatus == "submit"
|
|
586
|
+
// ? "#a5a6a8"
|
|
587
|
+
// : "#6938EF",
|
|
588
|
+
// };
|
|
589
|
+
// }
|
|
590
|
+
if (fixtureData.fixtureName === "space")
|
|
591
|
+
return;
|
|
592
|
+
const radius = 5;
|
|
593
|
+
const mainContainer = new fabric.Rect({
|
|
594
|
+
left: x,
|
|
595
|
+
top: y,
|
|
596
|
+
width: width,
|
|
597
|
+
height: height,
|
|
598
|
+
rx: radius,
|
|
599
|
+
ry: radius,
|
|
600
|
+
stroke: fixtureColor?.primary ?? "#51C1FF",
|
|
601
|
+
strokeWidth: 0.27,
|
|
602
|
+
fill: "white",
|
|
603
|
+
});
|
|
604
|
+
const topWidth = width / 1.05;
|
|
605
|
+
const topHeight = (15 / 100) * height;
|
|
606
|
+
const topX = x + (width - topWidth) / 2;
|
|
607
|
+
const topY = y + this.verticalMargin;
|
|
608
|
+
const topContainer = new fabric.Rect({
|
|
609
|
+
left: topX,
|
|
610
|
+
top: topY,
|
|
611
|
+
width: topWidth,
|
|
612
|
+
height: topHeight,
|
|
613
|
+
rx: radius,
|
|
614
|
+
ry: radius,
|
|
615
|
+
stroke: fixtureColor?.primary ?? "#51C1FF",
|
|
616
|
+
strokeWidth: 0.27,
|
|
617
|
+
fill: "white",
|
|
618
|
+
});
|
|
619
|
+
const innerBoxHeight = topHeight / 2;
|
|
620
|
+
const innerTopBox = new fabric.Path(`M ${topX + radius} ${topY}
|
|
621
|
+
L ${topX + topWidth - radius} ${topY}
|
|
622
|
+
Q ${topX + topWidth} ${topY} ${topX + topWidth} ${topY + radius}
|
|
623
|
+
L ${topX + topWidth} ${topY + innerBoxHeight}
|
|
624
|
+
L ${topX} ${topY + innerBoxHeight}
|
|
625
|
+
L ${topX} ${topY + radius}
|
|
626
|
+
Q ${topX} ${topY} ${topX + radius} ${topY} Z`, {
|
|
627
|
+
stroke: fixtureColor?.primary ?? "#51C1FF",
|
|
628
|
+
strokeWidth: 0.27,
|
|
629
|
+
fill: "transparent",
|
|
630
|
+
});
|
|
631
|
+
const topText = new fabric.Textbox(`Fixture ${fixtureData.associatedElementFixtureNumber
|
|
632
|
+
? fixtureData.associatedElementFixtureNumber +
|
|
633
|
+
(fixtureData?.fixtureNumber
|
|
634
|
+
? " - FX-" + fixtureData.fixtureNumber
|
|
635
|
+
: "")
|
|
636
|
+
: "new"}`, {
|
|
637
|
+
left: topX + topWidth / 2,
|
|
638
|
+
top: topY + innerBoxHeight / 2,
|
|
639
|
+
fontSize: 5,
|
|
640
|
+
fontFamily: "Inter",
|
|
641
|
+
fontWeight: "500",
|
|
642
|
+
fill: "#101828",
|
|
643
|
+
originX: "center",
|
|
644
|
+
originY: "center",
|
|
645
|
+
width: topWidth,
|
|
646
|
+
height: innerBoxHeight,
|
|
647
|
+
textAlign: "center",
|
|
648
|
+
});
|
|
649
|
+
const secondBoxY = topY + innerBoxHeight;
|
|
650
|
+
const innerBottomBox = new fabric.Path(`M ${topX} ${secondBoxY}
|
|
651
|
+
L ${topX + topWidth} ${secondBoxY}
|
|
652
|
+
L ${topX + topWidth} ${secondBoxY + innerBoxHeight - radius}
|
|
653
|
+
Q ${topX + topWidth} ${secondBoxY + innerBoxHeight} ${topX + topWidth - radius} ${secondBoxY + innerBoxHeight}
|
|
654
|
+
L ${topX + radius} ${secondBoxY + innerBoxHeight}
|
|
655
|
+
Q ${topX} ${secondBoxY + innerBoxHeight} ${topX} ${secondBoxY + innerBoxHeight - radius}
|
|
656
|
+
L ${topX} ${secondBoxY} Z`, {
|
|
657
|
+
fill: "#344054",
|
|
658
|
+
stroke: fixtureColor?.primary ?? "#51C1FF",
|
|
659
|
+
strokeWidth: 0.27,
|
|
660
|
+
});
|
|
661
|
+
let fontSize = 5;
|
|
662
|
+
let bottomText;
|
|
663
|
+
const textContent = `${fixtureData?.fixtureName} - ${fixtureData?.fixtureWidth.value} ${fixtureData?.fixtureWidth.unit}`;
|
|
664
|
+
const createBottomText = (fs) => new fabric.Textbox(textContent, {
|
|
665
|
+
left: topX + topWidth / 2,
|
|
666
|
+
top: secondBoxY + innerBoxHeight / 2,
|
|
667
|
+
fontSize: fs,
|
|
668
|
+
fontFamily: "Inter",
|
|
669
|
+
fontWeight: "500",
|
|
670
|
+
fill: "#FFFFFF",
|
|
671
|
+
originX: "center",
|
|
672
|
+
originY: "center",
|
|
673
|
+
textAlign: "center",
|
|
674
|
+
width: topWidth,
|
|
675
|
+
});
|
|
676
|
+
do {
|
|
677
|
+
bottomText = createBottomText(fontSize);
|
|
678
|
+
fontSize -= 0.5;
|
|
679
|
+
} while (bottomText.height > innerBoxHeight && fontSize > 2);
|
|
680
|
+
const midWidth = width / 1.05;
|
|
681
|
+
const midHeight = (77 / 100) * height;
|
|
682
|
+
const midX = x + (width - midWidth) / 2;
|
|
683
|
+
const midY = topY + topHeight + this.verticalMargin;
|
|
684
|
+
const midContainer = new fabric.Rect({
|
|
685
|
+
left: midX,
|
|
686
|
+
top: midY,
|
|
687
|
+
width: midWidth,
|
|
688
|
+
height: midHeight,
|
|
689
|
+
rx: radius,
|
|
690
|
+
ry: radius,
|
|
691
|
+
stroke: fixtureColor?.primary ?? "#51C1FF",
|
|
692
|
+
strokeWidth: 0.27,
|
|
693
|
+
fill: "white",
|
|
694
|
+
});
|
|
695
|
+
const shelfCount = fixtureData.shelfConfig?.length;
|
|
696
|
+
const boxHeight = midHeight / (shelfCount + 2);
|
|
697
|
+
const firstBoxY = midY;
|
|
698
|
+
const firstBox = new fabric.Path(`M ${midX + radius} ${firstBoxY}
|
|
699
|
+
L ${midX + midWidth - radius} ${firstBoxY}
|
|
700
|
+
Q ${midX + midWidth} ${firstBoxY} ${midX + midWidth} ${firstBoxY + radius}
|
|
701
|
+
L ${midX + midWidth} ${firstBoxY + boxHeight}
|
|
702
|
+
L ${midX} ${firstBoxY + boxHeight}
|
|
703
|
+
L ${midX} ${firstBoxY + radius}
|
|
704
|
+
Q ${midX} ${firstBoxY} ${midX + radius} ${firstBoxY} Z`, {
|
|
705
|
+
fill: fixtureColor?.fill
|
|
706
|
+
? fixtureColor?.fill
|
|
707
|
+
: fixtureData.issue
|
|
708
|
+
? "#D92D20"
|
|
709
|
+
: "#EAF8FF",
|
|
710
|
+
stroke: fixtureColor?.primary
|
|
711
|
+
? fixtureColor?.primary
|
|
712
|
+
: fixtureData.issue
|
|
713
|
+
? "#D92D20"
|
|
714
|
+
: "#6BCAFF",
|
|
715
|
+
strokeWidth: 0.27,
|
|
716
|
+
});
|
|
717
|
+
const firstBoxText = new fabric.Textbox(`${fixtureData?.header?.label ? fixtureData.header.label : ""}`, {
|
|
718
|
+
left: midX + midWidth / 2,
|
|
719
|
+
top: firstBoxY + boxHeight / 2,
|
|
720
|
+
fontSize: 7,
|
|
721
|
+
fontFamily: "Inter",
|
|
722
|
+
fontWeight: "600",
|
|
723
|
+
fill: fixtureData.issue ? "#fff" : "#101828",
|
|
724
|
+
originX: "center",
|
|
725
|
+
originY: "center",
|
|
726
|
+
width: midWidth,
|
|
727
|
+
height: boxHeight,
|
|
728
|
+
textAlign: "center",
|
|
729
|
+
});
|
|
730
|
+
const shelves = [];
|
|
731
|
+
const shelfHeight = midHeight - 2 * boxHeight;
|
|
732
|
+
if (fixtureData.productResolutionLevel === "L1") {
|
|
733
|
+
const boxY = firstBoxY + boxHeight;
|
|
734
|
+
const shelf = new fabric.Rect({
|
|
735
|
+
left: midX,
|
|
736
|
+
top: boxY,
|
|
737
|
+
width: midWidth,
|
|
738
|
+
height: shelfHeight,
|
|
739
|
+
// stroke: '#51C1FF',
|
|
740
|
+
stroke: fixtureColor?.primary
|
|
741
|
+
? fixtureColor?.primary
|
|
742
|
+
: fixtureData.issue
|
|
743
|
+
? "#D92D20"
|
|
744
|
+
: "#51C1FF",
|
|
745
|
+
strokeWidth: 0.27,
|
|
746
|
+
fill: "white",
|
|
747
|
+
});
|
|
748
|
+
const text = fixtureData?.productBrandName?.join(" + ") || "";
|
|
749
|
+
// Wrap text within fixture width and shrink to fit vertical space
|
|
750
|
+
const maxTextWidth = midWidth * 0.9;
|
|
751
|
+
const maxTextHeight = shelfHeight * 0.8;
|
|
752
|
+
let fontSizeL1 = 6;
|
|
753
|
+
const createWrappedTextbox = (fs) => new fabric.Textbox(text, {
|
|
754
|
+
left: midX + midWidth / 2,
|
|
755
|
+
top: boxY + shelfHeight / 2,
|
|
756
|
+
originX: "center",
|
|
757
|
+
originY: "center",
|
|
758
|
+
width: maxTextWidth,
|
|
759
|
+
fontSize: fs,
|
|
760
|
+
fontFamily: "Inter",
|
|
761
|
+
fontWeight: "500",
|
|
762
|
+
textAlign: "center",
|
|
763
|
+
fill: "#333",
|
|
764
|
+
selectable: false,
|
|
765
|
+
evented: false,
|
|
766
|
+
});
|
|
767
|
+
let wrappedText = createWrappedTextbox(fontSizeL1);
|
|
768
|
+
while (wrappedText.height > maxTextHeight && fontSizeL1 > 2) {
|
|
769
|
+
fontSizeL1 -= 0.5;
|
|
770
|
+
wrappedText = createWrappedTextbox(fontSizeL1);
|
|
771
|
+
}
|
|
772
|
+
const padding = 2;
|
|
773
|
+
const background = new fabric.Rect({
|
|
774
|
+
left: midX + midWidth / 2,
|
|
775
|
+
top: boxY + shelfHeight / 2,
|
|
776
|
+
originX: "center",
|
|
777
|
+
originY: "center",
|
|
778
|
+
width: Math.min(maxTextWidth, wrappedText.width) + padding,
|
|
779
|
+
height: wrappedText.getScaledHeight() + 2,
|
|
780
|
+
fill: "rgba(255, 255, 255, 0.5)",
|
|
781
|
+
selectable: false,
|
|
782
|
+
evented: false,
|
|
783
|
+
rx: 1,
|
|
784
|
+
ry: 1,
|
|
785
|
+
});
|
|
786
|
+
shelves.push(shelf, background, wrappedText);
|
|
787
|
+
}
|
|
788
|
+
else if (fixtureData.productResolutionLevel === "L2") {
|
|
789
|
+
for (let i = 0; i < shelfCount; i++) {
|
|
790
|
+
const boxY = firstBoxY + boxHeight * (i + 1);
|
|
791
|
+
const shelf = new fabric.Rect({
|
|
792
|
+
left: midX,
|
|
793
|
+
top: boxY,
|
|
794
|
+
width: midWidth,
|
|
795
|
+
height: boxHeight,
|
|
796
|
+
stroke: fixtureColor?.primary
|
|
797
|
+
? fixtureColor?.primary
|
|
798
|
+
: fixtureData.issue
|
|
799
|
+
? "#D92D20"
|
|
800
|
+
: "#51C1FF",
|
|
801
|
+
strokeWidth: 0.27,
|
|
802
|
+
fill: "white",
|
|
803
|
+
});
|
|
804
|
+
shelves.push(shelf);
|
|
805
|
+
if (fixtureData.shelfConfig?.[i].shelfType === "tray") {
|
|
806
|
+
const trayRows = fixtureData.shelfConfig?.[i].trayRows || 1;
|
|
807
|
+
const rowSpacing = boxHeight / trayRows;
|
|
808
|
+
for (let r = 1; r < trayRows; r++) {
|
|
809
|
+
const lineY = boxY + r * rowSpacing;
|
|
810
|
+
const trayLine = new fabric.Line([midX, lineY, midX + midWidth, lineY], {
|
|
811
|
+
stroke: "#999",
|
|
812
|
+
strokeWidth: 0.2,
|
|
813
|
+
selectable: false,
|
|
814
|
+
evented: false,
|
|
815
|
+
});
|
|
816
|
+
shelves.push(trayLine);
|
|
817
|
+
}
|
|
818
|
+
}
|
|
819
|
+
const text = fixtureData.shelfConfig?.[i].productBrandName?.join(" + ") || "";
|
|
820
|
+
// L2: Single-line ellipsis against full shelf width
|
|
821
|
+
const horizontalPadding = 6;
|
|
822
|
+
const maxTextWidth = Math.max(0, midWidth - horizontalPadding);
|
|
823
|
+
const maxTextHeight = boxHeight * 0.8;
|
|
824
|
+
const baseFontSize = 5;
|
|
825
|
+
const measure = (t) => new fabric.Text(t, {
|
|
826
|
+
fontSize: baseFontSize,
|
|
827
|
+
fontFamily: "Inter",
|
|
828
|
+
fontWeight: "500",
|
|
829
|
+
});
|
|
830
|
+
let displayText = text;
|
|
831
|
+
let measured = measure(displayText);
|
|
832
|
+
if ((measured.width ?? 0) > maxTextWidth) {
|
|
833
|
+
const ellipsis = "...";
|
|
834
|
+
let leftIdx = 0;
|
|
835
|
+
let rightIdx = text.length;
|
|
836
|
+
while (leftIdx < rightIdx) {
|
|
837
|
+
const midIdx = Math.floor((leftIdx + rightIdx) / 2);
|
|
838
|
+
const candidate = text.slice(0, midIdx) + ellipsis;
|
|
839
|
+
measured = measure(candidate);
|
|
840
|
+
if ((measured.width ?? 0) > maxTextWidth) {
|
|
841
|
+
rightIdx = midIdx;
|
|
842
|
+
}
|
|
843
|
+
else {
|
|
844
|
+
leftIdx = midIdx + 1;
|
|
845
|
+
}
|
|
846
|
+
}
|
|
847
|
+
displayText = text.slice(0, Math.max(0, leftIdx - 1)) + ellipsis;
|
|
848
|
+
measured = measure(displayText);
|
|
849
|
+
}
|
|
850
|
+
const background = new fabric.Rect({
|
|
851
|
+
left: midX + midWidth / 2,
|
|
852
|
+
top: boxY + boxHeight / 2,
|
|
853
|
+
originX: "center",
|
|
854
|
+
originY: "center",
|
|
855
|
+
width: midWidth,
|
|
856
|
+
height: Math.min(maxTextHeight, measured.height ?? 10) + 2,
|
|
857
|
+
fill: "rgba(255, 255, 255, 0.5)",
|
|
858
|
+
selectable: false,
|
|
859
|
+
evented: false,
|
|
860
|
+
rx: 1,
|
|
861
|
+
ry: 1,
|
|
862
|
+
});
|
|
863
|
+
const finalText = new fabric.Text(displayText, {
|
|
864
|
+
left: midX + midWidth / 2,
|
|
865
|
+
top: boxY + boxHeight / 2,
|
|
866
|
+
originX: "center",
|
|
867
|
+
originY: "center",
|
|
868
|
+
fontSize: baseFontSize,
|
|
869
|
+
fontFamily: "Inter",
|
|
870
|
+
fontWeight: "500",
|
|
871
|
+
fill: "#333",
|
|
872
|
+
selectable: false,
|
|
873
|
+
evented: false,
|
|
874
|
+
});
|
|
875
|
+
shelves.push(background, finalText);
|
|
876
|
+
}
|
|
877
|
+
}
|
|
878
|
+
else if (fixtureData.productResolutionLevel === "L3") {
|
|
879
|
+
const sequentialGroups = [];
|
|
880
|
+
let previousZone = null;
|
|
881
|
+
for (const shelf of fixtureData.shelfConfig) {
|
|
882
|
+
const zone = shelf.zone || "Unknown";
|
|
883
|
+
if (zone !== previousZone) {
|
|
884
|
+
sequentialGroups.push({ zone, shelves: [shelf] });
|
|
885
|
+
previousZone = zone;
|
|
886
|
+
}
|
|
887
|
+
else {
|
|
888
|
+
sequentialGroups[sequentialGroups.length - 1].shelves.push(shelf);
|
|
889
|
+
}
|
|
890
|
+
}
|
|
891
|
+
let currentShelfIndex = 0;
|
|
892
|
+
for (const group of sequentialGroups) {
|
|
893
|
+
const { zone, shelves: shelvesInGroup } = group;
|
|
894
|
+
const groupStartIndex = currentShelfIndex;
|
|
895
|
+
const groupLength = shelvesInGroup.length;
|
|
896
|
+
const groupTopY = firstBoxY + boxHeight * (groupStartIndex + 1);
|
|
897
|
+
const groupBottomY = groupTopY + boxHeight * groupLength;
|
|
898
|
+
for (let j = 0; j < shelvesInGroup.length; j++) {
|
|
899
|
+
const shelfData = shelvesInGroup[j];
|
|
900
|
+
const boxY = firstBoxY + boxHeight * (currentShelfIndex + 1);
|
|
901
|
+
const shelfRect = new fabric.Rect({
|
|
902
|
+
left: midX,
|
|
903
|
+
top: boxY,
|
|
904
|
+
width: midWidth,
|
|
905
|
+
height: boxHeight,
|
|
906
|
+
stroke: fixtureColor?.primary
|
|
907
|
+
? fixtureColor?.primary
|
|
908
|
+
: fixtureData.issue
|
|
909
|
+
? "#D92D20"
|
|
910
|
+
: "#51C1FF",
|
|
911
|
+
strokeWidth: 0.27,
|
|
912
|
+
fill: "white",
|
|
913
|
+
});
|
|
914
|
+
shelves.push(shelfRect);
|
|
915
|
+
if (shelfData.shelfType === "tray") {
|
|
916
|
+
const trayRows = shelfData.trayRows || 0;
|
|
917
|
+
const productPerShelf = shelfData.productPerShelf || 0;
|
|
918
|
+
const rowSpacing = boxHeight / trayRows;
|
|
919
|
+
const columnSpacing = midWidth / productPerShelf;
|
|
920
|
+
for (let r = 1; r < trayRows; r++) {
|
|
921
|
+
const lineY = boxY + r * rowSpacing;
|
|
922
|
+
const trayLine = new fabric.Line([midX, lineY, midX + midWidth, lineY], {
|
|
923
|
+
stroke: "#999",
|
|
924
|
+
strokeWidth: 0.2,
|
|
925
|
+
selectable: false,
|
|
926
|
+
evented: false,
|
|
927
|
+
});
|
|
928
|
+
shelves.push(trayLine);
|
|
929
|
+
}
|
|
930
|
+
for (let c = 1; c < productPerShelf; c++) {
|
|
931
|
+
const lineX = midX + c * columnSpacing;
|
|
932
|
+
const verticalLine = new fabric.Line([lineX, boxY, lineX, boxY + boxHeight], {
|
|
933
|
+
stroke: "#999",
|
|
934
|
+
strokeWidth: 0.2,
|
|
935
|
+
selectable: false,
|
|
936
|
+
evented: false,
|
|
937
|
+
});
|
|
938
|
+
shelves.push(verticalLine);
|
|
939
|
+
}
|
|
940
|
+
}
|
|
941
|
+
currentShelfIndex++;
|
|
942
|
+
}
|
|
943
|
+
const groupBottomLine = new fabric.Line([midX, groupBottomY, midX + midWidth, groupBottomY], {
|
|
944
|
+
stroke: fixtureColor?.primary
|
|
945
|
+
? fixtureColor?.primary
|
|
946
|
+
: fixtureData.issue
|
|
947
|
+
? "#D92D20"
|
|
948
|
+
: "#51C1FF",
|
|
949
|
+
strokeWidth: 1,
|
|
950
|
+
selectable: false,
|
|
951
|
+
evented: false,
|
|
952
|
+
});
|
|
953
|
+
shelves.push(groupBottomLine);
|
|
954
|
+
if (fixtureData.fixtureType === "wall") {
|
|
955
|
+
const labelPadding = 2;
|
|
956
|
+
const labelText = new fabric.Textbox(zone, {
|
|
957
|
+
left: midX + midWidth - labelPadding,
|
|
958
|
+
top: groupBottomY,
|
|
959
|
+
fontSize: 5,
|
|
960
|
+
fontFamily: "Inter",
|
|
961
|
+
fontWeight: "600",
|
|
962
|
+
fill: "#707070",
|
|
963
|
+
originX: "right",
|
|
964
|
+
originY: "bottom",
|
|
965
|
+
width: midWidth / 2,
|
|
966
|
+
textAlign: "right",
|
|
967
|
+
});
|
|
968
|
+
shelves.push(labelText);
|
|
969
|
+
}
|
|
970
|
+
const groupCenterX = midX + midWidth / 2;
|
|
971
|
+
const groupCenterY = groupTopY + (groupBottomY - groupTopY) / 2;
|
|
972
|
+
const sectionBrands = new Set();
|
|
973
|
+
shelvesInGroup?.forEach((shelf) => {
|
|
974
|
+
shelf.productBrandName?.forEach((brand) => {
|
|
975
|
+
sectionBrands.add(brand);
|
|
976
|
+
});
|
|
977
|
+
});
|
|
978
|
+
if ([...sectionBrands].length) {
|
|
979
|
+
const text = [...sectionBrands].join(" + ");
|
|
980
|
+
const padding = 2;
|
|
981
|
+
const maxTextWidth = midWidth;
|
|
982
|
+
const rectWidth = midWidth * 0.9;
|
|
983
|
+
const maxTextHeight = 20;
|
|
984
|
+
let fontSize = 6;
|
|
985
|
+
let wrappedText;
|
|
986
|
+
const createTextbox = (fs) => {
|
|
987
|
+
return new fabric.Textbox(text, {
|
|
988
|
+
fontSize: fs,
|
|
989
|
+
fontFamily: "Inter",
|
|
990
|
+
fontWeight: "400",
|
|
991
|
+
width: maxTextWidth,
|
|
992
|
+
textAlign: "center",
|
|
993
|
+
originX: "center",
|
|
994
|
+
originY: "center",
|
|
995
|
+
fill: "#101828",
|
|
996
|
+
selectable: false,
|
|
997
|
+
evented: false,
|
|
998
|
+
left: groupCenterX,
|
|
999
|
+
top: groupCenterY,
|
|
1000
|
+
});
|
|
1001
|
+
};
|
|
1002
|
+
do {
|
|
1003
|
+
wrappedText = createTextbox(fontSize);
|
|
1004
|
+
fontSize -= 0.5;
|
|
1005
|
+
} while (wrappedText.height > maxTextHeight && fontSize > 2);
|
|
1006
|
+
const rectHeight = wrappedText.getScaledHeight() + padding;
|
|
1007
|
+
const groupCenterText = new fabric.Rect({
|
|
1008
|
+
left: groupCenterX,
|
|
1009
|
+
top: groupCenterY,
|
|
1010
|
+
originX: "center",
|
|
1011
|
+
originY: "center",
|
|
1012
|
+
width: rectWidth,
|
|
1013
|
+
height: rectHeight,
|
|
1014
|
+
fill: "rgba(255, 255, 255, 0.5)",
|
|
1015
|
+
selectable: false,
|
|
1016
|
+
evented: false,
|
|
1017
|
+
rx: 1,
|
|
1018
|
+
ry: 1,
|
|
1019
|
+
});
|
|
1020
|
+
shelves.push(groupCenterText, wrappedText);
|
|
1021
|
+
}
|
|
1022
|
+
}
|
|
1023
|
+
}
|
|
1024
|
+
if (fixtureData?.vmConfig?.length) {
|
|
1025
|
+
for (let i = 0; i < fixtureData.vmConfig.length; i++) {
|
|
1026
|
+
const vm = fixtureData.vmConfig[i];
|
|
1027
|
+
const startIndex = vm.startYPosition - 1;
|
|
1028
|
+
const endIndex = vm.endYPosition - 1;
|
|
1029
|
+
const shelfTopY = firstBoxY + boxHeight * (startIndex + 1);
|
|
1030
|
+
const shelfHeightSpan = boxHeight * (endIndex - startIndex + 1);
|
|
1031
|
+
let vmHeightPx = shelfHeightSpan;
|
|
1032
|
+
let y = shelfTopY;
|
|
1033
|
+
if (vm.yZone && vm.yZone !== "stretch") {
|
|
1034
|
+
vmHeightPx = shelfHeightSpan / 3;
|
|
1035
|
+
switch (vm.yZone) {
|
|
1036
|
+
case "top":
|
|
1037
|
+
y = shelfTopY;
|
|
1038
|
+
break;
|
|
1039
|
+
case "middle":
|
|
1040
|
+
y = shelfTopY + vmHeightPx;
|
|
1041
|
+
break;
|
|
1042
|
+
case "bottom":
|
|
1043
|
+
y = shelfTopY + 2 * vmHeightPx;
|
|
1044
|
+
break;
|
|
1045
|
+
default:
|
|
1046
|
+
y = shelfTopY;
|
|
1047
|
+
}
|
|
1048
|
+
}
|
|
1049
|
+
let x = midX;
|
|
1050
|
+
let width = midWidth;
|
|
1051
|
+
if (!vm.xZone || vm.xZone === "stretch") {
|
|
1052
|
+
width = midWidth;
|
|
1053
|
+
x = midX;
|
|
1054
|
+
}
|
|
1055
|
+
else {
|
|
1056
|
+
const sectionWidth = midWidth / 3;
|
|
1057
|
+
switch (vm.xZone) {
|
|
1058
|
+
case "left":
|
|
1059
|
+
width = sectionWidth;
|
|
1060
|
+
x = midX;
|
|
1061
|
+
break;
|
|
1062
|
+
case "middle":
|
|
1063
|
+
width = sectionWidth;
|
|
1064
|
+
x = midX + sectionWidth;
|
|
1065
|
+
break;
|
|
1066
|
+
case "right":
|
|
1067
|
+
width = sectionWidth;
|
|
1068
|
+
x = midX + 2 * sectionWidth;
|
|
1069
|
+
break;
|
|
1070
|
+
default:
|
|
1071
|
+
width = midWidth;
|
|
1072
|
+
x = midX;
|
|
1073
|
+
}
|
|
1074
|
+
}
|
|
1075
|
+
if (vm.vmImageUrl) {
|
|
1076
|
+
try {
|
|
1077
|
+
const vmImage = await fabric.FabricImage.fromURL(`${this.cdnUrl}${vm.vmImageUrl}?v=${Date.now()}`, {
|
|
1078
|
+
crossOrigin: "anonymous",
|
|
1079
|
+
});
|
|
1080
|
+
vmImage.set({
|
|
1081
|
+
left: x + width / 2,
|
|
1082
|
+
top: y + vmHeightPx / 2,
|
|
1083
|
+
originX: "center",
|
|
1084
|
+
originY: "center",
|
|
1085
|
+
scaleX: width / vmImage.width,
|
|
1086
|
+
scaleY: vmHeightPx / vmImage.height,
|
|
1087
|
+
selectable: false,
|
|
1088
|
+
evented: false,
|
|
1089
|
+
});
|
|
1090
|
+
shelves.push(vmImage);
|
|
1091
|
+
}
|
|
1092
|
+
catch (err) {
|
|
1093
|
+
console.error(`Error loading image for VM: ${vm.vmImageUrl}`, err);
|
|
1094
|
+
}
|
|
1095
|
+
}
|
|
1096
|
+
else {
|
|
1097
|
+
const vmBox = new fabric.Rect({
|
|
1098
|
+
left: x,
|
|
1099
|
+
top: y,
|
|
1100
|
+
width,
|
|
1101
|
+
height: vmHeightPx,
|
|
1102
|
+
fill: fixtureColor?.vmFill ?? "#6938EF",
|
|
1103
|
+
selectable: false,
|
|
1104
|
+
evented: false,
|
|
1105
|
+
});
|
|
1106
|
+
let fontSize = 5;
|
|
1107
|
+
let vmText;
|
|
1108
|
+
const createVmText = (fs) => new fabric.Textbox(vm.vmName || "", {
|
|
1109
|
+
left: x + width / 2,
|
|
1110
|
+
top: y + vmHeightPx / 2,
|
|
1111
|
+
originX: "center",
|
|
1112
|
+
originY: "center",
|
|
1113
|
+
fontSize: fs,
|
|
1114
|
+
fontFamily: "Inter",
|
|
1115
|
+
fontWeight: "600",
|
|
1116
|
+
fill: "white",
|
|
1117
|
+
textAlign: "center",
|
|
1118
|
+
width: width,
|
|
1119
|
+
selectable: false,
|
|
1120
|
+
evented: false,
|
|
1121
|
+
});
|
|
1122
|
+
do {
|
|
1123
|
+
vmText = createVmText(fontSize);
|
|
1124
|
+
fontSize -= 0.5;
|
|
1125
|
+
} while (vmText.height > vmHeightPx && fontSize > 2);
|
|
1126
|
+
shelves.push(vmBox, vmText);
|
|
1127
|
+
}
|
|
1128
|
+
}
|
|
1129
|
+
}
|
|
1130
|
+
const lastBoxY = firstBoxY + boxHeight * (shelfCount + 1);
|
|
1131
|
+
const lastBox = new fabric.Path(`M ${midX} ${lastBoxY}
|
|
1132
|
+
L ${midX + midWidth} ${lastBoxY}
|
|
1133
|
+
L ${midX + midWidth} ${lastBoxY + boxHeight - radius}
|
|
1134
|
+
Q ${midX + midWidth} ${lastBoxY + boxHeight} ${midX + midWidth - radius} ${lastBoxY + boxHeight}
|
|
1135
|
+
L ${midX + radius} ${lastBoxY + boxHeight}
|
|
1136
|
+
Q ${midX} ${lastBoxY + boxHeight} ${midX} ${lastBoxY + boxHeight - radius}
|
|
1137
|
+
L ${midX} ${lastBoxY} Z`, {
|
|
1138
|
+
fill: fixtureColor?.fill
|
|
1139
|
+
? fixtureColor?.fill
|
|
1140
|
+
: fixtureData.issue
|
|
1141
|
+
? "#FEE4E2"
|
|
1142
|
+
: "#EAF8FF",
|
|
1143
|
+
stroke: fixtureColor?.primary
|
|
1144
|
+
? fixtureColor?.primary
|
|
1145
|
+
: fixtureData.issue
|
|
1146
|
+
? "#D92D20"
|
|
1147
|
+
: "#6BCAFF",
|
|
1148
|
+
strokeWidth: 0.27,
|
|
1149
|
+
});
|
|
1150
|
+
const lastBoxText = new fabric.Textbox(`${fixtureData?.footer?.label ? fixtureData.footer.label : ""}`, {
|
|
1151
|
+
left: midX + midWidth / 2,
|
|
1152
|
+
top: lastBoxY + boxHeight / 2,
|
|
1153
|
+
fontSize: 5,
|
|
1154
|
+
fontFamily: "Inter",
|
|
1155
|
+
fontWeight: "600",
|
|
1156
|
+
fill: "#101828",
|
|
1157
|
+
originX: "center",
|
|
1158
|
+
originY: "center",
|
|
1159
|
+
textAlign: "center",
|
|
1160
|
+
width: midWidth,
|
|
1161
|
+
heigth: boxHeight,
|
|
1162
|
+
});
|
|
1163
|
+
const bottomWidth = width / 1.05;
|
|
1164
|
+
const bottomHeight = (3 / 100) * height;
|
|
1165
|
+
const bottomX = x + (width - bottomWidth) / 2;
|
|
1166
|
+
const bottomY = midY + midHeight + this.verticalMargin;
|
|
1167
|
+
const bottomContainer = new fabric.Rect({
|
|
1168
|
+
left: bottomX,
|
|
1169
|
+
top: bottomY,
|
|
1170
|
+
width: bottomWidth,
|
|
1171
|
+
height: bottomHeight,
|
|
1172
|
+
rx: radius,
|
|
1173
|
+
ry: radius,
|
|
1174
|
+
stroke: "transparent",
|
|
1175
|
+
fill: "transparent",
|
|
1176
|
+
});
|
|
1177
|
+
const bottomTextContent = new fabric.Textbox(`Shelves - ${fixtureData.shelfConfig?.length} . Products - ${fixtureData.fixtureCapacity} . VM - ${fixtureData.vmConfig?.length}`, {
|
|
1178
|
+
left: bottomX + bottomWidth / 2,
|
|
1179
|
+
top: bottomY + bottomHeight / 2,
|
|
1180
|
+
fontSize: 5,
|
|
1181
|
+
fontFamily: "Inter",
|
|
1182
|
+
fontWeight: "500",
|
|
1183
|
+
fill: "#333",
|
|
1184
|
+
originX: "center",
|
|
1185
|
+
originY: "center",
|
|
1186
|
+
textAlign: "center",
|
|
1187
|
+
width: bottomWidth,
|
|
1188
|
+
height: bottomHeight,
|
|
1189
|
+
});
|
|
1190
|
+
const fixtureGroup = new fabric.Group([
|
|
1191
|
+
mainContainer,
|
|
1192
|
+
topContainer,
|
|
1193
|
+
innerTopBox,
|
|
1194
|
+
topText,
|
|
1195
|
+
innerBottomBox,
|
|
1196
|
+
bottomText,
|
|
1197
|
+
midContainer,
|
|
1198
|
+
firstBox,
|
|
1199
|
+
firstBoxText,
|
|
1200
|
+
...shelves,
|
|
1201
|
+
lastBox,
|
|
1202
|
+
lastBoxText,
|
|
1203
|
+
bottomContainer,
|
|
1204
|
+
bottomTextContent,
|
|
1205
|
+
], {
|
|
1206
|
+
left: x,
|
|
1207
|
+
top: y,
|
|
1208
|
+
evented: true,
|
|
1209
|
+
hoverCursor: "pointer",
|
|
1210
|
+
lockScalingX: true,
|
|
1211
|
+
lockScalingY: true,
|
|
1212
|
+
subTargetCheck: true,
|
|
1213
|
+
// hasControls: fixtureData.fixtureType === "floor",
|
|
1214
|
+
// lockRotation: fixtureData.fixtureType !== "floor",
|
|
1215
|
+
// lockMovementX: currentFixtureInfo?.wallDirection === 'up' || currentFixtureInfo?.wallDirection === 'down',
|
|
1216
|
+
// lockMovementY: currentFixtureInfo?.wallDirection === 'right' || currentFixtureInfo?.wallDirection === 'left',
|
|
1217
|
+
angle: fixtureData.relativePosition?.angle ?? 0,
|
|
1218
|
+
});
|
|
1219
|
+
fixtureGroup.fixtureType = fixtureData.fixtureType;
|
|
1220
|
+
fixtureGroup.wallIndex = fixtureData.associatedElementNumber;
|
|
1221
|
+
fixtureGroup.fixtureIndex = fixtureData.associatedElementFixtureNumber;
|
|
1222
|
+
fixtureGroup.fixtureId = fixtureData._id;
|
|
1223
|
+
fixtureGroup.issue = fixtureData.issue;
|
|
1224
|
+
fixtureGroup.issueData = fixtureData.issueData;
|
|
1225
|
+
fixtureGroup.taskType = fixtureData.taskType;
|
|
1226
|
+
fixtureGroup.vmConfig = fixtureData.vmConfig;
|
|
1227
|
+
fixtureGroup.objType = "fixture";
|
|
1228
|
+
canvas.add(fixtureGroup);
|
|
1229
|
+
// If newly added, focus it once
|
|
1230
|
+
if (fixtureData?.newAdded) {
|
|
1231
|
+
try {
|
|
1232
|
+
canvas.setActiveObject(fixtureGroup);
|
|
1233
|
+
canvas.requestRenderAll();
|
|
1234
|
+
}
|
|
1235
|
+
catch { }
|
|
1236
|
+
delete fixtureData.newAdded;
|
|
1237
|
+
}
|
|
1238
|
+
return fixtureGroup;
|
|
1239
|
+
}
|
|
1240
|
+
drawOtherElementsWithLabel(canvas, x, y, a, width, height, element) {
|
|
1241
|
+
const rect = new fabric.Rect({
|
|
1242
|
+
width,
|
|
1243
|
+
height,
|
|
1244
|
+
rx: 2,
|
|
1245
|
+
ry: 2,
|
|
1246
|
+
fill: "#DAF1FF",
|
|
1247
|
+
stroke: "#6BCAFF",
|
|
1248
|
+
strokeWidth: 2,
|
|
1249
|
+
originX: "center",
|
|
1250
|
+
originY: "center",
|
|
1251
|
+
});
|
|
1252
|
+
const text = new fabric.Textbox(element.fixtureName, {
|
|
1253
|
+
width: width - 8,
|
|
1254
|
+
fontSize: 14,
|
|
1255
|
+
fontFamily: "Inter",
|
|
1256
|
+
fontWeight: "500",
|
|
1257
|
+
fill: "#101828",
|
|
1258
|
+
originX: "center",
|
|
1259
|
+
originY: "center",
|
|
1260
|
+
textAlign: "center",
|
|
1261
|
+
editable: false,
|
|
1262
|
+
splitByGrapheme: true,
|
|
1263
|
+
});
|
|
1264
|
+
const textBounds = text.getBoundingRect();
|
|
1265
|
+
if (textBounds.height > height - 8) {
|
|
1266
|
+
const scale = (height - 8) / textBounds.height;
|
|
1267
|
+
text.set("fontSize", text.fontSize * scale);
|
|
1268
|
+
}
|
|
1269
|
+
const group = new fabric.Group([rect, text], {
|
|
1270
|
+
left: x,
|
|
1271
|
+
top: y,
|
|
1272
|
+
angle: a,
|
|
1273
|
+
originX: "center",
|
|
1274
|
+
originY: "center",
|
|
1275
|
+
});
|
|
1276
|
+
group.objType = "others";
|
|
1277
|
+
group.data = element;
|
|
1278
|
+
canvas.add(group);
|
|
1279
|
+
if (element?.newAdded) {
|
|
1280
|
+
try {
|
|
1281
|
+
canvas.setActiveObject(group);
|
|
1282
|
+
canvas.requestRenderAll();
|
|
1283
|
+
}
|
|
1284
|
+
catch { }
|
|
1285
|
+
delete element.newAdded;
|
|
1286
|
+
}
|
|
1287
|
+
return group;
|
|
1288
|
+
}
|
|
1289
|
+
highlightFixture(canvas, group, blink = false) {
|
|
1290
|
+
this.removeHighlight(canvas);
|
|
1291
|
+
this.cancelHighlightBlink = false;
|
|
1292
|
+
const center = group.getCenterPoint();
|
|
1293
|
+
const realWidth = group.width * group.scaleX;
|
|
1294
|
+
const realHeight = group.height * group.scaleY;
|
|
1295
|
+
const rect = new fabric.Rect({
|
|
1296
|
+
left: center.x,
|
|
1297
|
+
top: center.y,
|
|
1298
|
+
width: realWidth + 8,
|
|
1299
|
+
height: realHeight + 8,
|
|
1300
|
+
rx: 5,
|
|
1301
|
+
ry: 5,
|
|
1302
|
+
stroke: "#00aaff",
|
|
1303
|
+
strokeWidth: 3,
|
|
1304
|
+
fill: "transparent",
|
|
1305
|
+
selectable: false,
|
|
1306
|
+
evented: false,
|
|
1307
|
+
originX: "center",
|
|
1308
|
+
originY: "center",
|
|
1309
|
+
angle: group.angle || 0,
|
|
1310
|
+
opacity: blink ? 0 : 1,
|
|
1311
|
+
});
|
|
1312
|
+
rect.data = { type: "fixture-highlight" };
|
|
1313
|
+
canvas.add(rect);
|
|
1314
|
+
canvas.moveObjectTo(rect, canvas.getObjects().length - 1);
|
|
1315
|
+
canvas.requestRenderAll();
|
|
1316
|
+
this.activeHighlight = rect;
|
|
1317
|
+
// --- No blinking needed ---
|
|
1318
|
+
if (!blink)
|
|
1319
|
+
return;
|
|
1320
|
+
let cycles = 0;
|
|
1321
|
+
const blinkOnce = () => {
|
|
1322
|
+
if (this.cancelHighlightBlink) {
|
|
1323
|
+
this.removeHighlight(canvas);
|
|
1324
|
+
return;
|
|
1325
|
+
}
|
|
1326
|
+
rect.animate({ opacity: 1 }, {
|
|
1327
|
+
duration: 200,
|
|
1328
|
+
onChange: () => canvas.requestRenderAll(),
|
|
1329
|
+
onComplete: () => {
|
|
1330
|
+
if (this.cancelHighlightBlink) {
|
|
1331
|
+
this.removeHighlight(canvas);
|
|
1332
|
+
return;
|
|
1333
|
+
}
|
|
1334
|
+
rect.animate({ opacity: 0 }, {
|
|
1335
|
+
duration: 200,
|
|
1336
|
+
onChange: () => canvas.requestRenderAll(),
|
|
1337
|
+
onComplete: () => {
|
|
1338
|
+
if (this.cancelHighlightBlink) {
|
|
1339
|
+
this.removeHighlight(canvas);
|
|
1340
|
+
return;
|
|
1341
|
+
}
|
|
1342
|
+
cycles++;
|
|
1343
|
+
if (cycles < 20) {
|
|
1344
|
+
blinkOnce();
|
|
1345
|
+
}
|
|
1346
|
+
else {
|
|
1347
|
+
if (!this.cancelHighlightBlink) {
|
|
1348
|
+
rect.set({ opacity: 1 });
|
|
1349
|
+
canvas.requestRenderAll();
|
|
1350
|
+
}
|
|
1351
|
+
}
|
|
1352
|
+
},
|
|
1353
|
+
});
|
|
1354
|
+
},
|
|
1355
|
+
});
|
|
1356
|
+
};
|
|
1357
|
+
blinkOnce();
|
|
1358
|
+
}
|
|
1359
|
+
removeHighlight(canvas) {
|
|
1360
|
+
this.cancelHighlightBlink = true;
|
|
1361
|
+
canvas.getObjects().forEach((obj) => {
|
|
1362
|
+
if (obj.data?.type === "fixture-highlight") {
|
|
1363
|
+
canvas.remove(obj);
|
|
1364
|
+
}
|
|
1365
|
+
});
|
|
1366
|
+
this.activeHighlight = null;
|
|
1367
|
+
canvas.requestRenderAll();
|
|
1368
|
+
}
|
|
1369
|
+
async applyStores() {
|
|
1370
|
+
const value = this.form.value.stores;
|
|
1371
|
+
if (!value)
|
|
1372
|
+
return;
|
|
1373
|
+
const codes = value
|
|
1374
|
+
.split(',')
|
|
1375
|
+
.map((s) => s.trim().toUpperCase())
|
|
1376
|
+
.filter(Boolean);
|
|
1377
|
+
const response = await lastValueFrom(this.apiService.getPlanoId({ stores: codes }));
|
|
1378
|
+
const data = Array.isArray(response.data) ? response.data : [];
|
|
1379
|
+
const foundStoreSet = new Set(data.map((store) => store.storeName.toUpperCase()));
|
|
1380
|
+
const foundChips = data.map((store) => ({
|
|
1381
|
+
storeCode: store.storeName,
|
|
1382
|
+
status: 'pending',
|
|
1383
|
+
planoId: store._id
|
|
1384
|
+
}));
|
|
1385
|
+
const notFoundChips = codes
|
|
1386
|
+
.filter((code) => !foundStoreSet.has(code))
|
|
1387
|
+
.map((code) => ({
|
|
1388
|
+
storeCode: code,
|
|
1389
|
+
status: 'not_found'
|
|
1390
|
+
}));
|
|
1391
|
+
this.storeChips = [...foundChips, ...notFoundChips];
|
|
1392
|
+
const activeChip = this.storeChips.find((chip) => chip.status === 'pending');
|
|
1393
|
+
this.selectedStore = activeChip?.storeCode;
|
|
1394
|
+
this.getStoreFixtures(activeChip.planoId);
|
|
1395
|
+
}
|
|
1396
|
+
delay(ms) {
|
|
1397
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
1398
|
+
}
|
|
1399
|
+
async autoApprove() {
|
|
1400
|
+
for (let i = 0; i < this.storeChips.length; i++) {
|
|
1401
|
+
const { storeCode, status, planoId } = this.storeChips[i];
|
|
1402
|
+
if (status === 'not_found')
|
|
1403
|
+
continue;
|
|
1404
|
+
const planoRes = await lastValueFrom(this.apiService.getStoreFixtures({ id: [planoId] }));
|
|
1405
|
+
if (!planoRes?.data?.[0]?.floors?.length) {
|
|
1406
|
+
return;
|
|
1407
|
+
}
|
|
1408
|
+
this.planoData = structuredClone(planoRes.data[0]);
|
|
1409
|
+
this.floorData = structuredClone(planoRes.data[0].floors[0]);
|
|
1410
|
+
this.renderFloor(this.beforeCanvas);
|
|
1411
|
+
this.renderFloor(this.afterCanvas);
|
|
1412
|
+
this.loading = false;
|
|
1413
|
+
setTimeout(() => {
|
|
1414
|
+
this.resizeCanvas(this.beforeCanvas, this.beforeContainerRef);
|
|
1415
|
+
this.fitCanvasToLayoutAnimated(this.beforeCanvas);
|
|
1416
|
+
this.resizeCanvas(this.afterCanvas, this.afterContainerRef);
|
|
1417
|
+
this.fitCanvasToLayoutAnimated(this.afterCanvas);
|
|
1418
|
+
}, 100);
|
|
1419
|
+
const payload = {
|
|
1420
|
+
...this.form.value,
|
|
1421
|
+
stores: [storeCode]
|
|
1422
|
+
};
|
|
1423
|
+
const aiRes = await lastValueFrom(this.apiService.findFixtureAi(payload));
|
|
1424
|
+
if (aiRes.code === 200) {
|
|
1425
|
+
this.updateAiResponseFixture(aiRes.data);
|
|
1426
|
+
this.renderFloor(this.afterCanvas);
|
|
1427
|
+
setTimeout(() => {
|
|
1428
|
+
const actionFixture = this.afterCanvas
|
|
1429
|
+
.getObjects()
|
|
1430
|
+
.find((obj) => {
|
|
1431
|
+
return obj.objType === "fixture" && obj.fixtureId === aiRes.data._id;
|
|
1432
|
+
});
|
|
1433
|
+
if (!actionFixture)
|
|
1434
|
+
return;
|
|
1435
|
+
this.highlightFixture(this.afterCanvas, actionFixture, true);
|
|
1436
|
+
}, 0);
|
|
1437
|
+
}
|
|
1438
|
+
const updatePayload = {
|
|
1439
|
+
fixtures: [this.selectedFixture]
|
|
1440
|
+
};
|
|
1441
|
+
await lastValueFrom(this.apiService.updateFixtureData(updatePayload));
|
|
1442
|
+
this.storeChips[i].status = 'approved';
|
|
1443
|
+
const activeChip = this.storeChips.find((chip) => chip.status === 'pending');
|
|
1444
|
+
this.selectedStore = activeChip?.storeCode;
|
|
1445
|
+
await this.delay(3000);
|
|
1446
|
+
}
|
|
1447
|
+
}
|
|
1448
|
+
approveStore() {
|
|
1449
|
+
const payload = {
|
|
1450
|
+
fixtures: [this.selectedFixture]
|
|
1451
|
+
};
|
|
1452
|
+
this.apiService.updateFixtureData(payload).subscribe({
|
|
1453
|
+
next: (res) => {
|
|
1454
|
+
if (res.code === 200) {
|
|
1455
|
+
this.storeChips.find((chip) => chip.storeCode === this.selectedStore).status = 'approved';
|
|
1456
|
+
const nextChip = this.storeChips.find((chip) => chip.status === 'pending');
|
|
1457
|
+
this.selectedStore = nextChip?.storeCode;
|
|
1458
|
+
this.getStoreFixtures(nextChip.planoId);
|
|
1459
|
+
}
|
|
1460
|
+
}
|
|
1461
|
+
});
|
|
1462
|
+
}
|
|
1463
|
+
selectStore(chip) {
|
|
1464
|
+
this.selectedStore = chip.storeCode;
|
|
1465
|
+
this.getStoreFixtures(chip.planoId);
|
|
1466
|
+
}
|
|
1467
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CollectionUpdateAiComponent, deps: [{ token: i1.FormBuilder }, { token: i2.StoreBuilderService }, { token: i3.ToastService }, { token: i4.ActivatedRoute }, { token: i5.TitleCasePipe }], target: i0.ɵɵFactoryTarget.Component });
|
|
1468
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: CollectionUpdateAiComponent, isStandalone: true, selector: "lib-collection-update-ai", providers: [TitleCasePipe], viewQueries: [{ propertyName: "beforeCanvasRef", first: true, predicate: ["beforeCanvas"], descendants: true }, { propertyName: "beforeContainerRef", first: true, predicate: ["beforeCanvasContainer"], descendants: true }, { propertyName: "afterCanvasRef", first: true, predicate: ["afterCanvas"], descendants: true }, { propertyName: "afterContainerRef", first: true, predicate: ["afterCanvasContainer"], descendants: true }], ngImport: i0, template: "<form class=\"ai-container\" [formGroup]=\"form\" (ngSubmit)=\"submitPrompt()\">\r\n\r\n <!-- ================= Applicable Stores ================= -->\r\n <div class=\"field\">\r\n <label>Applicable Stores</label>\r\n\r\n <div class=\"d-flex gap-2 align-items-center\">\r\n <input type=\"text\" class=\"form-control\" formControlName=\"stores\" placeholder=\"e.g. LKST345, LKST346\" />\r\n\r\n <button type=\"button\" class=\"btn btn-outline\" (click)=\"applyStores()\">\r\n Apply\r\n </button>\r\n </div>\r\n\r\n <span class=\"error\" *ngIf=\"form.controls.stores.touched && form.controls.stores.invalid\">\r\n Applicable stores is required\r\n </span>\r\n </div>\r\n\r\n <!-- ================= Store Chips ================= -->\r\n <div class=\"store-chips\" *ngIf=\"storeChips.length\">\r\n <div (click)=\"selectStore(chip)\" class=\"store-chip\" *ngFor=\"let chip of storeChips\" [ngClass]=\"[\r\n chip.status,\r\n chip.storeCode === selectedStore ? 'active' : ''\r\n ]\">\r\n {{ chip.storeCode }}\r\n </div>\r\n </div>\r\n\r\n\r\n <!-- ================= Update Type ================= -->\r\n <div class=\"field\">\r\n <label>Update Type</label>\r\n\r\n <div class=\"radio-group\">\r\n <label class=\"radio-option\">\r\n <input type=\"radio\" formControlName=\"updateType\" value=\"vm\" />\r\n VM\r\n </label>\r\n\r\n <label class=\"radio-option\">\r\n <input type=\"radio\" formControlName=\"updateType\" value=\"pid\" />\r\n PID\r\n </label>\r\n </div>\r\n\r\n <span class=\"error\" *ngIf=\"form.controls.updateType.touched && form.controls.updateType.invalid\">\r\n Update type is required\r\n </span>\r\n </div>\r\n\r\n <!-- ================= AI Prompt ================= -->\r\n <div class=\"field\">\r\n <label>AI Prompt</label>\r\n\r\n <div class=\"row\">\r\n\r\n <!-- Selection Criteria -->\r\n <div class=\"col-md-6\">\r\n <div class=\"form-group\">\r\n <label class=\"small fw-semibold\">Selection Criteria</label>\r\n\r\n <textarea rows=\"3\" class=\"form-control\" formControlName=\"selectionCriteria\"\r\n placeholder=\"e.g. Select premium shades on wall fixtures\"></textarea>\r\n\r\n <span class=\"error\"\r\n *ngIf=\"form.controls.selectionCriteria.touched && form.controls.selectionCriteria.invalid\">\r\n Selection criteria is required\r\n </span>\r\n </div>\r\n </div>\r\n\r\n <!-- Updation Criteria -->\r\n <div class=\"col-md-6\">\r\n <div class=\"form-group\">\r\n <label class=\"small fw-semibold\">Updation Criteria</label>\r\n\r\n <textarea rows=\"3\" class=\"form-control\" formControlName=\"updationCriteria\"\r\n placeholder=\"e.g. Move selected fixtures to mid shelves\"></textarea>\r\n\r\n <span class=\"error\" *ngIf=\"form.controls.updationCriteria.touched && form.controls.updationCriteria.invalid\">\r\n Updation criteria is required\r\n </span>\r\n </div>\r\n </div>\r\n\r\n </div>\r\n </div>\r\n\r\n <!-- ================= Submit ================= -->\r\n <div class=\"d-flex justify-content-end mt-3\">\r\n <button type=\"submit\" class=\"submit-btn\" [disabled]=\"loading\">\r\n {{ loading ? 'Thinking\u2026' : 'Submit' }}\r\n </button>\r\n </div>\r\n\r\n <!-- ================= Canvases ================= -->\r\n <div class=\"canvas-grid mt-4\">\r\n\r\n <div class=\"canvas-box\" #beforeCanvasContainer>\r\n <div class=\"canvas-title\">Before Update</div>\r\n <canvas #beforeCanvas></canvas>\r\n </div>\r\n\r\n <div class=\"canvas-box\" #afterCanvasContainer>\r\n <div class=\"canvas-title\">After Update</div>\r\n <canvas #afterCanvas></canvas>\r\n </div>\r\n\r\n </div>\r\n\r\n <div class=\"d-flex align-items-center justify-content-end\">\r\n <!-- ================= Auto Approve ================= -->\r\n <button (click)=\"autoApprove()\" type=\"button\" class=\"btn btn-warning me-3\">\r\n \u26A0 Auto Approve\r\n </button>\r\n\r\n <!-- ================= Approve Changes ================= -->\r\n <button type=\"button\" class=\"btn btn-success\" (click)=\"approveStore()\" [disabled]=\"form.invalid\">\r\n \u2714 Approve Changes\r\n </button>\r\n </div>\r\n\r\n\r\n</form>", styles: ["@charset \"UTF-8\";.ai-container{display:flex;flex-direction:column;gap:16px;height:100%}.field{display:flex;flex-direction:column;gap:6px}.field label{font-size:13px;font-weight:600;color:#374151}.field input,.field textarea{padding:10px;font-size:14px;border-radius:6px;border:1px solid #d1d5db}.error{font-size:12px;color:#dc2626}.submit-btn{width:140px;padding:10px;background:#2563eb;color:#fff;border:none;border-radius:6px;cursor:pointer}.submit-btn:disabled{background:#9ca3af}.canvas-grid{display:grid;grid-template-columns:1fr 1fr;gap:16px}.canvas-box{border:1px solid #e5e7eb;border-radius:8px;padding:8px;background:#f9fafb;display:flex;flex-direction:column;align-items:center;height:70dvh}.canvas-title{font-size:14px;font-weight:600;margin-bottom:8px}canvas{background:#fff;border-radius:4px}.approve-bar{display:flex;justify-content:flex-end;margin-top:auto}.approve-btn{padding:12px 20px;background:#16a34a;color:#fff;border:none;border-radius:6px;font-size:14px;cursor:pointer}.approve-btn:disabled{background:#9ca3af}.radio-group{display:flex;gap:20px;margin-top:6px}.radio-option{display:flex;align-items:center;gap:6px;cursor:pointer}.store-chips{display:flex;flex-wrap:wrap;gap:8px;margin-top:10px}.store-chip{padding:6px 12px;border-radius:14px;font-size:13px;font-weight:600;color:#fff;cursor:pointer}.store-chip.pending{background-color:#6c757d}.store-chip.approved{background-color:#198754}.store-chip.skipped{background-color:#0d6efd}.store-chip.not_found{background-color:#dc3545}.store-chip.active{animation:chip-pulse 1.4s infinite ease-in-out;z-index:2}.store-chip.active:after{content:\"\";position:absolute;inset:-3px;border-radius:16px;border:2px solid #20c997;box-shadow:0 0 6px #20c997b3,0 0 12px #20c99766;animation:ring-pulse 1.4s infinite ease-in-out;pointer-events:none}@keyframes chip-pulse{0%{transform:scale(1)}50%{transform:scale(1.025)}to{transform:scale(1)}}@keyframes ring-pulse{0%{opacity:.4}50%{opacity:.9}to{opacity:.4}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { 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.RadioControlValueAccessor, selector: "input[type=radio][formControlName],input[type=radio][formControl],input[type=radio][ngModel]", inputs: ["name", "formControlName", "value"] }, { 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.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }] });
|
|
1469
|
+
}
|
|
1470
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CollectionUpdateAiComponent, decorators: [{
|
|
1471
|
+
type: Component,
|
|
1472
|
+
args: [{ selector: 'lib-collection-update-ai', standalone: true, imports: [CommonModule, ReactiveFormsModule], providers: [TitleCasePipe], template: "<form class=\"ai-container\" [formGroup]=\"form\" (ngSubmit)=\"submitPrompt()\">\r\n\r\n <!-- ================= Applicable Stores ================= -->\r\n <div class=\"field\">\r\n <label>Applicable Stores</label>\r\n\r\n <div class=\"d-flex gap-2 align-items-center\">\r\n <input type=\"text\" class=\"form-control\" formControlName=\"stores\" placeholder=\"e.g. LKST345, LKST346\" />\r\n\r\n <button type=\"button\" class=\"btn btn-outline\" (click)=\"applyStores()\">\r\n Apply\r\n </button>\r\n </div>\r\n\r\n <span class=\"error\" *ngIf=\"form.controls.stores.touched && form.controls.stores.invalid\">\r\n Applicable stores is required\r\n </span>\r\n </div>\r\n\r\n <!-- ================= Store Chips ================= -->\r\n <div class=\"store-chips\" *ngIf=\"storeChips.length\">\r\n <div (click)=\"selectStore(chip)\" class=\"store-chip\" *ngFor=\"let chip of storeChips\" [ngClass]=\"[\r\n chip.status,\r\n chip.storeCode === selectedStore ? 'active' : ''\r\n ]\">\r\n {{ chip.storeCode }}\r\n </div>\r\n </div>\r\n\r\n\r\n <!-- ================= Update Type ================= -->\r\n <div class=\"field\">\r\n <label>Update Type</label>\r\n\r\n <div class=\"radio-group\">\r\n <label class=\"radio-option\">\r\n <input type=\"radio\" formControlName=\"updateType\" value=\"vm\" />\r\n VM\r\n </label>\r\n\r\n <label class=\"radio-option\">\r\n <input type=\"radio\" formControlName=\"updateType\" value=\"pid\" />\r\n PID\r\n </label>\r\n </div>\r\n\r\n <span class=\"error\" *ngIf=\"form.controls.updateType.touched && form.controls.updateType.invalid\">\r\n Update type is required\r\n </span>\r\n </div>\r\n\r\n <!-- ================= AI Prompt ================= -->\r\n <div class=\"field\">\r\n <label>AI Prompt</label>\r\n\r\n <div class=\"row\">\r\n\r\n <!-- Selection Criteria -->\r\n <div class=\"col-md-6\">\r\n <div class=\"form-group\">\r\n <label class=\"small fw-semibold\">Selection Criteria</label>\r\n\r\n <textarea rows=\"3\" class=\"form-control\" formControlName=\"selectionCriteria\"\r\n placeholder=\"e.g. Select premium shades on wall fixtures\"></textarea>\r\n\r\n <span class=\"error\"\r\n *ngIf=\"form.controls.selectionCriteria.touched && form.controls.selectionCriteria.invalid\">\r\n Selection criteria is required\r\n </span>\r\n </div>\r\n </div>\r\n\r\n <!-- Updation Criteria -->\r\n <div class=\"col-md-6\">\r\n <div class=\"form-group\">\r\n <label class=\"small fw-semibold\">Updation Criteria</label>\r\n\r\n <textarea rows=\"3\" class=\"form-control\" formControlName=\"updationCriteria\"\r\n placeholder=\"e.g. Move selected fixtures to mid shelves\"></textarea>\r\n\r\n <span class=\"error\" *ngIf=\"form.controls.updationCriteria.touched && form.controls.updationCriteria.invalid\">\r\n Updation criteria is required\r\n </span>\r\n </div>\r\n </div>\r\n\r\n </div>\r\n </div>\r\n\r\n <!-- ================= Submit ================= -->\r\n <div class=\"d-flex justify-content-end mt-3\">\r\n <button type=\"submit\" class=\"submit-btn\" [disabled]=\"loading\">\r\n {{ loading ? 'Thinking\u2026' : 'Submit' }}\r\n </button>\r\n </div>\r\n\r\n <!-- ================= Canvases ================= -->\r\n <div class=\"canvas-grid mt-4\">\r\n\r\n <div class=\"canvas-box\" #beforeCanvasContainer>\r\n <div class=\"canvas-title\">Before Update</div>\r\n <canvas #beforeCanvas></canvas>\r\n </div>\r\n\r\n <div class=\"canvas-box\" #afterCanvasContainer>\r\n <div class=\"canvas-title\">After Update</div>\r\n <canvas #afterCanvas></canvas>\r\n </div>\r\n\r\n </div>\r\n\r\n <div class=\"d-flex align-items-center justify-content-end\">\r\n <!-- ================= Auto Approve ================= -->\r\n <button (click)=\"autoApprove()\" type=\"button\" class=\"btn btn-warning me-3\">\r\n \u26A0 Auto Approve\r\n </button>\r\n\r\n <!-- ================= Approve Changes ================= -->\r\n <button type=\"button\" class=\"btn btn-success\" (click)=\"approveStore()\" [disabled]=\"form.invalid\">\r\n \u2714 Approve Changes\r\n </button>\r\n </div>\r\n\r\n\r\n</form>", styles: ["@charset \"UTF-8\";.ai-container{display:flex;flex-direction:column;gap:16px;height:100%}.field{display:flex;flex-direction:column;gap:6px}.field label{font-size:13px;font-weight:600;color:#374151}.field input,.field textarea{padding:10px;font-size:14px;border-radius:6px;border:1px solid #d1d5db}.error{font-size:12px;color:#dc2626}.submit-btn{width:140px;padding:10px;background:#2563eb;color:#fff;border:none;border-radius:6px;cursor:pointer}.submit-btn:disabled{background:#9ca3af}.canvas-grid{display:grid;grid-template-columns:1fr 1fr;gap:16px}.canvas-box{border:1px solid #e5e7eb;border-radius:8px;padding:8px;background:#f9fafb;display:flex;flex-direction:column;align-items:center;height:70dvh}.canvas-title{font-size:14px;font-weight:600;margin-bottom:8px}canvas{background:#fff;border-radius:4px}.approve-bar{display:flex;justify-content:flex-end;margin-top:auto}.approve-btn{padding:12px 20px;background:#16a34a;color:#fff;border:none;border-radius:6px;font-size:14px;cursor:pointer}.approve-btn:disabled{background:#9ca3af}.radio-group{display:flex;gap:20px;margin-top:6px}.radio-option{display:flex;align-items:center;gap:6px;cursor:pointer}.store-chips{display:flex;flex-wrap:wrap;gap:8px;margin-top:10px}.store-chip{padding:6px 12px;border-radius:14px;font-size:13px;font-weight:600;color:#fff;cursor:pointer}.store-chip.pending{background-color:#6c757d}.store-chip.approved{background-color:#198754}.store-chip.skipped{background-color:#0d6efd}.store-chip.not_found{background-color:#dc3545}.store-chip.active{animation:chip-pulse 1.4s infinite ease-in-out;z-index:2}.store-chip.active:after{content:\"\";position:absolute;inset:-3px;border-radius:16px;border:2px solid #20c997;box-shadow:0 0 6px #20c997b3,0 0 12px #20c99766;animation:ring-pulse 1.4s infinite ease-in-out;pointer-events:none}@keyframes chip-pulse{0%{transform:scale(1)}50%{transform:scale(1.025)}to{transform:scale(1)}}@keyframes ring-pulse{0%{opacity:.4}50%{opacity:.9}to{opacity:.4}}\n"] }]
|
|
1473
|
+
}], ctorParameters: () => [{ type: i1.FormBuilder }, { type: i2.StoreBuilderService }, { type: i3.ToastService }, { type: i4.ActivatedRoute }, { type: i5.TitleCasePipe }], propDecorators: { beforeCanvasRef: [{
|
|
1474
|
+
type: ViewChild,
|
|
1475
|
+
args: ['beforeCanvas']
|
|
1476
|
+
}], beforeContainerRef: [{
|
|
1477
|
+
type: ViewChild,
|
|
1478
|
+
args: ["beforeCanvasContainer"]
|
|
1479
|
+
}], afterCanvasRef: [{
|
|
1480
|
+
type: ViewChild,
|
|
1481
|
+
args: ['afterCanvas']
|
|
1482
|
+
}], afterContainerRef: [{
|
|
1483
|
+
type: ViewChild,
|
|
1484
|
+
args: ["afterCanvasContainer"]
|
|
1485
|
+
}] } });
|
|
1486
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collection-update-ai.component.js","sourceRoot":"","sources":["../../../../../../projects/tango-store-builder/src/lib/components/collection-update-ai/collection-update-ai.component.ts","../../../../../../projects/tango-store-builder/src/lib/components/collection-update-ai/collection-update-ai.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ViewChild, ElementRef, OnDestroy, AfterViewInit } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { ReactiveFormsModule, FormBuilder, FormGroup, Validators } from '@angular/forms';\r\nimport { lastValueFrom, Subject, takeUntil } from 'rxjs';\r\nimport { StoreBuilderService } from '../../services/store-builder.service';\r\nimport { ToastService } from 'tango-app-ui-shared';\r\nimport * as fabric from \"fabric\";\r\nimport { ActivatedRoute } from '@angular/router';\r\nimport { TitleCasePipe } from \"@angular/common\";\r\n\r\ntype StoreStatus = 'pending' | 'approved' | 'skipped' | 'not_found';\r\n\r\ninterface StoreChip {\r\n  storeCode: string;\r\n  planoId: string;\r\n  status: StoreStatus;\r\n}\r\n@Component({\r\n  selector: 'lib-collection-update-ai',\r\n  templateUrl: './collection-update-ai.component.html',\r\n  styleUrl: './collection-update-ai.component.scss',\r\n  standalone: true,\r\n  imports: [CommonModule, ReactiveFormsModule],\r\n  providers: [TitleCasePipe],\r\n})\r\n\r\nexport class CollectionUpdateAiComponent implements OnDestroy, AfterViewInit {\r\n\r\n  form: FormGroup;\r\n  loading = false;\r\n\r\n  @ViewChild('beforeCanvas') beforeCanvasRef!: ElementRef<HTMLCanvasElement>;\r\n  beforeCanvas!: fabric.Canvas;\r\n  @ViewChild(\"beforeCanvasContainer\") beforeContainerRef!: ElementRef<HTMLDivElement>;\r\n  @ViewChild('afterCanvas') afterCanvasRef!: ElementRef<HTMLCanvasElement>;\r\n  afterCanvas!: fabric.Canvas;\r\n  @ViewChild(\"afterCanvasContainer\") afterContainerRef!: ElementRef<HTMLDivElement>;\r\n  destroy$ = new Subject();\r\n  planoData: any;\r\n  floorData: any;\r\n  wallThickness = 30;\r\n  verticalMargin = (1.25 / 100) * this.toPixels(1524, \"mm\");\r\n  cdnUrl: string;\r\n  activeHighlight: fabric.Rect | null = null;\r\n  cancelHighlightBlink = false;\r\n  storeChips: StoreChip[] = [];\r\n  selectedStore?: string;\r\n  selectedFixture: any\r\n\r\n\r\n\r\n\r\n  constructor(private fb: FormBuilder,\r\n    private apiService: StoreBuilderService,\r\n    private toastService: ToastService,\r\n    private route: ActivatedRoute,\r\n    private titleCase: TitleCasePipe,\r\n\r\n  ) {\r\n    this.form = this.fb.group({\r\n      stores: ['', Validators.required],\r\n      updateType: ['vm', Validators.required],\r\n      selectionCriteria: ['', Validators.required],\r\n      updationCriteria: ['', Validators.required],\r\n    });\r\n  }\r\n\r\n  ngAfterViewInit(): void {\r\n    this.route.queryParams\r\n      .pipe(takeUntil(this.destroy$))\r\n      .subscribe((params) => {\r\n        if (params?.planoId) {\r\n          this.getStoreFixtures(params?.planoId);\r\n          if (params?.floorId) {\r\n            this.floorData = { _id: params?.floorId };\r\n          }\r\n        }\r\n      });\r\n\r\n    this.beforeCanvas = new fabric.Canvas(this.beforeCanvasRef?.nativeElement, {\r\n      selectionFullyContained: true,\r\n      perPixelTargetFind: true,\r\n      targetFindTolerance: 5,\r\n    });\r\n\r\n    this.afterCanvas = new fabric.Canvas(this.afterCanvasRef?.nativeElement, {\r\n      selectionFullyContained: true,\r\n      perPixelTargetFind: true,\r\n      targetFindTolerance: 5,\r\n    });\r\n\r\n\r\n    this.setupZoomHandlers(this.beforeCanvas);\r\n    this.setupZoomHandlers(this.afterCanvas);\r\n\r\n  }\r\n\r\n  setupZoomHandlers(canvas: fabric.Canvas): void {\r\n    canvas.on(\"mouse:wheel\", (opt: fabric.TEvent<WheelEvent>) => {\r\n      const e = opt.e;\r\n\r\n      if (e.ctrlKey || e.metaKey) {\r\n        const delta = e.deltaY;\r\n        let zoom = canvas.getZoom();\r\n        zoom *= 0.995 ** delta;\r\n\r\n        zoom = Math.max(0.2, Math.min(zoom, 3));\r\n\r\n        canvas.zoomToPoint(\r\n          {\r\n            x: e.offsetX,\r\n            y: e.offsetY,\r\n          } as fabric.Point,\r\n          zoom\r\n        );\r\n\r\n        e.preventDefault();\r\n        e.stopPropagation();\r\n      }\r\n    });\r\n\r\n    let isDragging = false;\r\n    let lastPosX = 0;\r\n    let lastPosY = 0;\r\n\r\n    canvas.on(\"mouse:down\", (opt: any) => {\r\n      if (!opt.target) {\r\n        isDragging = true;\r\n        canvas.selection = false;\r\n        const e = opt.e;\r\n        lastPosX = e.clientX;\r\n        lastPosY = e.clientY;\r\n      }\r\n    });\r\n\r\n    canvas.on(\"mouse:move\", (opt: any) => {\r\n      if (isDragging && canvas.viewportTransform) {\r\n        const e = opt.e;\r\n        const vpt = canvas.viewportTransform;\r\n        vpt[4] += e.clientX - lastPosX;\r\n        vpt[5] += e.clientY - lastPosY;\r\n        canvas.requestRenderAll();\r\n        lastPosX = e.clientX;\r\n        lastPosY = e.clientY;\r\n      }\r\n    });\r\n\r\n    canvas.on(\"mouse:up\", () => {\r\n      isDragging = false;\r\n      canvas.selection = true;\r\n    });\r\n  }\r\n\r\n  updateAiResponseFixture(fixture: any) {\r\n    let matchFound = false\r\n    for (let i = 0; i < this.floorData.layoutPolygon.length; i++) {\r\n      const element = this.floorData.layoutPolygon[i];\r\n\r\n      if (element.elementType === 'wall' && Array.isArray(element.fixtures)) {\r\n        for (let j = 0; j < element.fixtures.length; j++) {\r\n          const fx = element.fixtures[j];\r\n\r\n          if (fx._id === fixture._id) {\r\n            this.floorData.layoutPolygon[i].fixtures[j] = {\r\n              ...fixture\r\n            };\r\n            matchFound = true\r\n            this.selectedFixture = fixture\r\n            break;\r\n          }\r\n        }\r\n      }\r\n    }\r\n\r\n    if (matchFound) return;\r\n\r\n    for (let i = 0; i < this.floorData?.centerFixture.length; i++) {\r\n      const fx = this.floorData?.centerFixture[i];\r\n\r\n      if (fx._id === fixture._id) {\r\n        this.floorData.centerFixture[i] = {\r\n          ...fixture\r\n        }\r\n        this.selectedFixture = fixture\r\n        break;\r\n      }\r\n    }\r\n\r\n\r\n  }\r\n\r\n  submitPrompt() {\r\n    if (this.form.invalid) {\r\n      this.form.markAllAsTouched();\r\n      return;\r\n    }\r\n\r\n    this.loading = true;\r\n\r\n    const payload = {\r\n      ...this.form.value,\r\n      stores: [this.selectedStore]\r\n    };\r\n\r\n    this.apiService.findFixtureAi(payload).subscribe({\r\n      next: (res: any) => {\r\n        this.loading = false;\r\n\r\n        if (res.code === 200) {\r\n          this.updateAiResponseFixture(res.data)\r\n          this.renderFloor(this.afterCanvas)\r\n\r\n\r\n          setTimeout(() => {\r\n            const actionFixture = this.afterCanvas\r\n              .getObjects()\r\n              .find(\r\n                (obj: any) => {\r\n                  return obj.objType === \"fixture\" && obj.fixtureId === res.data._id\r\n                }\r\n              ) as fabric.Group | undefined;\r\n\r\n            if (!actionFixture) return;\r\n            this.highlightFixture(this.afterCanvas, actionFixture, true);\r\n          }, 0);\r\n        }\r\n      },\r\n      error: () => {\r\n        this.loading = false;\r\n      },\r\n    });\r\n  }\r\n\r\n\r\n  async getStoreFixtures(storeId: string) {\r\n    this.loading = true;\r\n    this.apiService\r\n      .getStoreFixtures({ id: [storeId] })\r\n      .pipe(takeUntil(this.destroy$))\r\n      .subscribe({\r\n        next: async (res: any) => {\r\n          if (!res?.data?.[0]?.floors?.length) {\r\n            return;\r\n          }\r\n\r\n          this.planoData = structuredClone(res.data[0]);\r\n\r\n          this.floorData = structuredClone(res.data[0].floors[0]);\r\n\r\n          this.renderFloor(this.beforeCanvas);\r\n          this.renderFloor(this.afterCanvas);\r\n\r\n          this.loading = false;\r\n          setTimeout(() => {\r\n            this.resizeCanvas(this.beforeCanvas, this.beforeContainerRef);\r\n            this.fitCanvasToLayoutAnimated(this.beforeCanvas);\r\n\r\n            this.resizeCanvas(this.afterCanvas, this.afterContainerRef);\r\n            this.fitCanvasToLayoutAnimated(this.afterCanvas);\r\n          }, 300);\r\n        },\r\n        error: (err: any) => {\r\n          this.loading = false;\r\n          this.toastService.getErrorToast(\"Failed to fetch plano\");\r\n        },\r\n      });\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.destroy$.next(true);\r\n    this.destroy$.complete();\r\n  }\r\n\r\n  resizeCanvas(canvas: fabric.Canvas, conteiner: ElementRef<HTMLDivElement>): void {\r\n    const container = conteiner.nativeElement;\r\n    canvas.setDimensions({\r\n      width: container.clientWidth - 50,\r\n\r\n      height: container.clientHeight - 40,\r\n    });\r\n    canvas.requestRenderAll();\r\n  }\r\n\r\n  private fitCanvasToLayoutAnimated(canvas: fabric.Canvas) {\r\n    if (!canvas) return;\r\n\r\n    const bounds = this.getFullLayoutBounds(canvas);\r\n    if (!bounds.width || !bounds.height) return;\r\n\r\n    const canvasW = canvas.getWidth();\r\n    const canvasH = canvas.getHeight();\r\n\r\n    // Final zoom to fit (with padding)\r\n    const scaleX = canvasW / bounds.width;\r\n    const scaleY = canvasH / bounds.height;\r\n    const targetZoom = Math.min(scaleX, scaleY) * 0.9; // padding 10%\r\n\r\n    // Desired viewport offset to center layout\r\n    const targetVpt = [\r\n      ...(canvas.viewportTransform || [1, 0, 0, 1, 0, 0]),\r\n    ] as fabric.TMat2D;\r\n\r\n    targetVpt[0] = targetZoom; // scaleX\r\n    targetVpt[3] = targetZoom; // scaleY\r\n\r\n    targetVpt[4] = canvasW / 2 - (bounds.left + bounds.width / 2) * targetZoom; // center X\r\n    targetVpt[5] = canvasH / 2 - (bounds.top + bounds.height / 2) * targetZoom; // center Y\r\n\r\n    // Current viewport (starting point of animation)\r\n    const startVpt = [...canvas.viewportTransform!] as fabric.TMat2D;\r\n\r\n    fabric.util.animate({\r\n      startValue: 0,\r\n      endValue: 1,\r\n      duration: 100,\r\n      easing: fabric.util.ease.easeOutCubic,\r\n      onChange: (value: number) => {\r\n        const vpt = canvas.viewportTransform!;\r\n\r\n        // Interpolate matrix values\r\n        vpt[0] = startVpt[0] + (targetVpt[0] - startVpt[0]) * value;\r\n        vpt[3] = startVpt[3] + (targetVpt[3] - startVpt[3]) * value;\r\n        vpt[4] = startVpt[4] + (targetVpt[4] - startVpt[4]) * value;\r\n        vpt[5] = startVpt[5] + (targetVpt[5] - startVpt[5]) * value;\r\n\r\n        canvas.requestRenderAll();\r\n      },\r\n    });\r\n  }\r\n\r\n  private getFullLayoutBounds(canvas: fabric.Canvas) {\r\n    const objects = canvas.getObjects();\r\n    if (!objects.length) return { left: 0, top: 0, width: 0, height: 0 };\r\n\r\n    let minX = Infinity,\r\n      minY = Infinity,\r\n      maxX = -Infinity,\r\n      maxY = -Infinity;\r\n\r\n    for (const obj of objects) {\r\n      const rect = obj.getBoundingRect();\r\n      minX = Math.min(minX, rect.left);\r\n      minY = Math.min(minY, rect.top);\r\n      maxX = Math.max(maxX, rect.left + rect.width);\r\n      maxY = Math.max(maxY, rect.top + rect.height);\r\n    }\r\n\r\n    return {\r\n      left: minX,\r\n      top: minY,\r\n      width: maxX - minX,\r\n      height: maxY - minY,\r\n    };\r\n  }\r\n\r\n  renderFloor(canvas: fabric.Canvas) {\r\n    this.drawInitialContent(canvas);\r\n  }\r\n\r\n  async drawInitialContent(canvas: fabric.Canvas): Promise<void> {\r\n    canvas.clear();\r\n    await this.drawLayoutElements(canvas);\r\n    // this.drawBg();\r\n  }\r\n\r\n  async drawLayoutElements(canvas: fabric.Canvas) {\r\n    for (let index = 0; index < this.floorData.layoutPolygon.length; index++) {\r\n      const element = this.floorData.layoutPolygon[index];\r\n\r\n      let wall;\r\n\r\n      element.relativePosition ??= {\r\n        x: 0,\r\n        y: 0,\r\n        unit: \"ft\",\r\n      };\r\n\r\n      const x = this.toPixels(element.relativePosition.x, element.unit);\r\n      const y = this.toPixels(element.relativePosition.y, element.unit);\r\n\r\n      if (element.elementType === \"wall\") {\r\n        const height = this.wallThickness;\r\n        const width = this.toPixels(element.distance, element.unit);\r\n        wall = await this.drawWall(canvas, x, y, height, width, element);\r\n      } else if (element.elementType === \"entrance\") {\r\n        const arrowWidth = 80;\r\n        const arrowHeight = 75;\r\n        const arrowBase = 40;\r\n        this.drawEntrance(\r\n          canvas,\r\n          x,\r\n          y,\r\n          arrowWidth,\r\n          arrowHeight,\r\n          arrowBase,\r\n          element\r\n        );\r\n      }\r\n\r\n      if (element.elementType === \"wall\") {\r\n        const fixtureGroups: fabric.Group[] = [];\r\n\r\n        for (let j = 0; j < element?.fixtures?.length; j++) {\r\n          const fixture = element.fixtures[j];\r\n          const x = this.toPixels(\r\n            fixture.relativePosition.x,\r\n            fixture.relativePosition.unit\r\n          );\r\n          const y = this.toPixels(\r\n            fixture.relativePosition.y,\r\n            fixture.relativePosition.unit\r\n          );\r\n          const fixtureHeight = this.toPixels(\r\n            fixture.fixtureStaticLength.value,\r\n            fixture.fixtureStaticLength.unit\r\n          );\r\n          const fixtureWidth = this.toPixels(\r\n            fixture.fixtureStaticWidth.value,\r\n            fixture.fixtureStaticWidth.unit\r\n          );\r\n\r\n          const fixtureObj = await this.drawFixture(\r\n            canvas,\r\n            x,\r\n            y,\r\n            fixtureHeight,\r\n            fixtureWidth,\r\n            fixture\r\n          );\r\n          fixtureGroups.push(fixtureObj as fabric.Group);\r\n        }\r\n\r\n        if (element.isLocked && wall) {\r\n          canvas.remove(wall);\r\n          fixtureGroups.forEach((f) => canvas.remove(f));\r\n\r\n          const combinedGroup = new fabric.Group([wall, ...fixtureGroups], {\r\n            hasControls: true,\r\n            selectable: true,\r\n            subTargetCheck: true,\r\n          });\r\n\r\n          combinedGroup.set({ isLocked: true });\r\n          canvas.add(combinedGroup);\r\n        }\r\n\r\n        for (let j = 0; j < element?.otherElements?.length; j++) {\r\n          const otherElement = element.otherElements[j];\r\n          const x = this.toPixels(otherElement.relativePosition.x, \"ft\");\r\n          const y = this.toPixels(otherElement.relativePosition.y, \"ft\");\r\n          const a = otherElement.relativePosition.angle;\r\n          const h = this.toPixels(\r\n            otherElement.fixtureStaticLength.value,\r\n            otherElement.fixtureStaticLength.unit\r\n          );\r\n          const w = this.toPixels(\r\n            otherElement.fixtureStaticWidth.value,\r\n            otherElement.fixtureStaticWidth.unit\r\n          );\r\n\r\n          const fixtureObj = this.drawOtherElementsWithLabel(\r\n            canvas,\r\n            x,\r\n            y,\r\n            a,\r\n            w,\r\n            h,\r\n            otherElement\r\n          );\r\n\r\n          // elementsGroup.push(fixtureObj as fabric.Group);\r\n        }\r\n      }\r\n    }\r\n\r\n    for (let i = 0; i < this.floorData?.centerFixture?.length; i++) {\r\n      const fixture = this.floorData.centerFixture[i];\r\n      const x = this.toPixels(\r\n        fixture.relativePosition.x,\r\n        fixture.relativePosition.unit\r\n      );\r\n      const y = this.toPixels(\r\n        fixture.relativePosition.y,\r\n        fixture.relativePosition.unit\r\n      );\r\n      const fixtureHeight = this.toPixels(\r\n        fixture.fixtureStaticLength.value,\r\n        fixture.fixtureStaticLength.unit\r\n      );\r\n      const fixtureWidth = this.toPixels(\r\n        fixture.fixtureStaticWidth.value,\r\n        fixture.fixtureStaticWidth.unit\r\n      );\r\n      await this.drawFixture(\r\n        canvas,\r\n        x,\r\n        y,\r\n        fixtureHeight,\r\n        fixtureWidth,\r\n        fixture\r\n      );\r\n    }\r\n\r\n    for (let j = 0; j < this.floorData?.otherElements?.length; j++) {\r\n      const otherElement = this.floorData.otherElements[j];\r\n      const x = this.toPixels(otherElement.relativePosition.x, \"ft\");\r\n      const y = this.toPixels(otherElement.relativePosition.y, \"ft\");\r\n      const a = otherElement.relativePosition.angle;\r\n      const h = this.toPixels(\r\n        otherElement.fixtureStaticLength.value,\r\n        otherElement.fixtureStaticLength.unit\r\n      );\r\n      const w = this.toPixels(\r\n        otherElement.fixtureStaticWidth.value,\r\n        otherElement.fixtureStaticWidth.unit\r\n      );\r\n\r\n      const fixtureObj = this.drawOtherElementsWithLabel(\r\n        canvas,\r\n        x,\r\n        y,\r\n        a,\r\n        w,\r\n        h,\r\n        otherElement\r\n      );\r\n    }\r\n\r\n    return true;\r\n  }\r\n\r\n\r\n  toPixels(value: number, unit: string, dpi = 2.8) {\r\n    let pixels: number;\r\n\r\n    if (unit === \"mm\") {\r\n      pixels = (value / 25.4) * dpi;\r\n    } else if (unit === \"ft\") {\r\n      pixels = value * 12 * dpi;\r\n    } else {\r\n      throw new Error(\"Unsupported unit. Use 'mm' or 'ft'.\");\r\n    }\r\n\r\n    return Number(pixels.toFixed(4));\r\n  }\r\n\r\n  async drawWall(\r\n    canvas: fabric.Canvas,\r\n    x: number,\r\n    y: number,\r\n    height: number,\r\n    width: number,\r\n    elementData: any\r\n  ) {\r\n    const radius = 5;\r\n    const fillColor = \"#F6FCFF\";\r\n    const strokeColor = \"#D0D5DD\";\r\n    const labelColor = \"#009BF3\";\r\n    let text = `${this.titleCase.transform(elementData.elementType)} ${elementData.elementNumber\r\n      }`;\r\n\r\n    const wall = new fabric.Rect({\r\n      left: 0,\r\n      top: 0,\r\n      width,\r\n      height,\r\n      rx: radius,\r\n      ry: radius,\r\n      fill: fillColor,\r\n      stroke: strokeColor,\r\n      strokeWidth: 0.27,\r\n      selectable: false,\r\n      strokeUniform: true,\r\n    });\r\n\r\n    const centerX = width / 2;\r\n    const centerY = height / 2;\r\n    const isVertical = height > width;\r\n\r\n    const label = new fabric.FabricText(text, {\r\n      left: centerX,\r\n      top: centerY,\r\n      fontSize: 10,\r\n      fontFamily: \"Inter\",\r\n      fontWeight: \"500\",\r\n      fill: labelColor,\r\n      originX: \"center\",\r\n      originY: \"center\",\r\n      angle: isVertical ? -90 : 0,\r\n      selectable: false,\r\n    });\r\n\r\n    const arrowHead = 5;\r\n    const elements: fabric.Object[] = [wall, label];\r\n\r\n    if (isVertical) {\r\n      const textWidth = label.width || 0;\r\n      const topArrowY = 5;\r\n      const topArrowEndY = centerY - textWidth / 2 - 5;\r\n\r\n      if (topArrowEndY > topArrowY + arrowHead) {\r\n        elements.push(\r\n          new fabric.Line([centerX, topArrowY, centerX, topArrowEndY], {\r\n            stroke: labelColor,\r\n            strokeWidth: 1,\r\n            selectable: false,\r\n          }),\r\n          new fabric.Triangle({\r\n            left: centerX,\r\n            top: topArrowY,\r\n            width: arrowHead,\r\n            height: arrowHead,\r\n            angle: 0,\r\n            fill: labelColor,\r\n            originX: \"center\",\r\n            originY: \"bottom\",\r\n            selectable: false,\r\n          })\r\n        );\r\n      }\r\n\r\n      const bottomArrowStartY = centerY + textWidth / 2 + 5;\r\n      const bottomArrowEndY = height - 5;\r\n\r\n      if (bottomArrowEndY > bottomArrowStartY + arrowHead) {\r\n        elements.push(\r\n          new fabric.Line(\r\n            [centerX, bottomArrowStartY, centerX, bottomArrowEndY],\r\n            {\r\n              stroke: labelColor,\r\n              strokeWidth: 1,\r\n              selectable: false,\r\n            }\r\n          ),\r\n          new fabric.Triangle({\r\n            left: centerX,\r\n            top: bottomArrowEndY,\r\n            width: arrowHead,\r\n            height: arrowHead,\r\n            angle: 180,\r\n            fill: labelColor,\r\n            originX: \"center\",\r\n            originY: \"top\",\r\n            selectable: false,\r\n          })\r\n        );\r\n      }\r\n    } else {\r\n      const textWidth = label.width || 0;\r\n      const leftArrowStartX = 5;\r\n      const leftArrowEndX = centerX - textWidth / 2 - 5;\r\n\r\n      if (leftArrowEndX > leftArrowStartX + arrowHead) {\r\n        elements.push(\r\n          new fabric.Line([leftArrowStartX, centerY, leftArrowEndX, centerY], {\r\n            stroke: labelColor,\r\n            strokeWidth: 1,\r\n            selectable: false,\r\n          }),\r\n          new fabric.Triangle({\r\n            left: leftArrowStartX,\r\n            top: centerY,\r\n            width: arrowHead,\r\n            height: arrowHead,\r\n            angle: -90,\r\n            fill: labelColor,\r\n            originX: \"center\",\r\n            originY: \"center\",\r\n            selectable: false,\r\n          })\r\n        );\r\n      }\r\n\r\n      const rightArrowStartX = centerX + textWidth / 2 + 5;\r\n      const rightArrowEndX = width - 5;\r\n\r\n      if (rightArrowEndX > rightArrowStartX + arrowHead) {\r\n        elements.push(\r\n          new fabric.Line(\r\n            [rightArrowStartX, centerY, rightArrowEndX, centerY],\r\n            {\r\n              stroke: labelColor,\r\n              strokeWidth: 1,\r\n              selectable: false,\r\n            }\r\n          ),\r\n          new fabric.Triangle({\r\n            left: rightArrowEndX,\r\n            top: centerY,\r\n            width: arrowHead,\r\n            height: arrowHead,\r\n            angle: 90,\r\n            fill: labelColor,\r\n            originX: \"center\",\r\n            originY: \"center\",\r\n            selectable: false,\r\n          })\r\n        );\r\n      }\r\n    }\r\n\r\n\r\n    const group = new fabric.Group(elements, {\r\n      left: x,\r\n      top: y,\r\n      hasControls: true,\r\n      lockScalingX: isVertical,\r\n      lockScalingY: !isVertical,\r\n      // lockRotation: true,\r\n      evented: true,\r\n      hoverCursor: \"pointer\",\r\n      angle: elementData.angle,\r\n      subTargetCheck: true,\r\n    }) as fabric.Group & {\r\n      objType: string;\r\n      data: any;\r\n    };\r\n\r\n    group.objType = elementData.elementType;\r\n    group.data = elementData;\r\n\r\n\r\n    canvas.add(group);\r\n\r\n    // If newly added, focus it once\r\n    if ((elementData as any)?.newAdded) {\r\n      try {\r\n        canvas.setActiveObject(group);\r\n        canvas.requestRenderAll();\r\n      } catch { }\r\n      delete (elementData as any).newAdded;\r\n    }\r\n\r\n    return group;\r\n  }\r\n\r\n  drawEntrance(\r\n    canvas: fabric.Canvas,\r\n    x: number,\r\n    y: number,\r\n    width: number,\r\n    height: number,\r\n    arrowWidth: number,\r\n    elementData: any\r\n  ) {\r\n    const bodyWidth = width - arrowWidth;\r\n\r\n    const pathData = [\r\n      `M 0 0`,\r\n      `L ${bodyWidth} 0`,\r\n      `L ${bodyWidth} ${-height / 2}`,\r\n      `L ${width} ${height / 2}`,\r\n      `L ${bodyWidth} ${height + height / 2}`,\r\n      `L ${bodyWidth} ${height}`,\r\n      `L 0 ${height}`,\r\n      `Z`,\r\n    ].join(\" \");\r\n\r\n    const arrow = new fabric.Path(pathData, {\r\n      fill: \"#03A9F4\",\r\n      selectable: false,\r\n      evented: false,\r\n    });\r\n\r\n    const maxTextHeight = height - 6;\r\n    const text = new fabric.Text(\"Entrance\", {\r\n      fontFamily: \"Arial\",\r\n      fontWeight: \"bold\",\r\n      fill: \"white\",\r\n      angle: -90,\r\n      originX: \"center\",\r\n      originY: \"center\",\r\n      left: 15,\r\n      top: height / 2,\r\n      fontSize: 20,\r\n      selectable: false,\r\n      evented: false,\r\n    });\r\n\r\n    const textBounds = text.getBoundingRect();\r\n    if (textBounds.height > maxTextHeight) {\r\n      const scale = maxTextHeight / textBounds.height;\r\n      text.set(\"fontSize\", text.fontSize! * scale);\r\n    }\r\n\r\n    const group = new fabric.Group([arrow, text], {\r\n      left: x,\r\n      top: y,\r\n      lockScalingX: true,\r\n      lockScalingY: true,\r\n      lockScalingFlip: true,\r\n      angle: elementData.angle,\r\n    }) as fabric.Group & {\r\n      objType: string;\r\n      data: any;\r\n    };\r\n\r\n    group.objType = elementData.elementType;\r\n    group.data = elementData;\r\n\r\n    canvas.add(group);\r\n  }\r\n\r\n  async drawFixture(\r\n    canvas: fabric.Canvas,\r\n    x: number,\r\n    y: number,\r\n    height: number,\r\n    width: number,\r\n    fixtureData: any\r\n  ) {\r\n    let fixtureColor: any;\r\n    // if ([\"fixture\", \"fixture-redo\"].includes(this.publishingState)) {\r\n    //   fixtureColor = {\r\n    //     primary:\r\n    //       fixtureData.status == \"\" && fixtureData.taskStatus == \"submit\"\r\n    //         ? \"#a5a6a8\"\r\n    //         : fixtureData.status == \"agree\"\r\n    //         ? \"#6CE9A6\"\r\n    //         : \"#51C1FF\",\r\n    //     fill:\r\n    //       fixtureData.status == \"\" && fixtureData.taskStatus == \"submit\"\r\n    //         ? \"#f3f4f8\"\r\n    //         : fixtureData.status == \"agree\"\r\n    //         ? \"#6CE9A6\"\r\n    //         : \"#EAF8FF\",\r\n    //     vmFill:\r\n    //       fixtureData.status == \"\" && fixtureData.taskStatus == \"submit\"\r\n    //         ? \"#a5a6a8\"\r\n    //         : \"#6938EF\",\r\n    //   };\r\n    // }\r\n    if (fixtureData.fixtureName === \"space\") return;\r\n    const radius = 5;\r\n    const mainContainer = new fabric.Rect({\r\n      left: x,\r\n      top: y,\r\n      width: width,\r\n      height: height,\r\n      rx: radius,\r\n      ry: radius,\r\n      stroke: fixtureColor?.primary ?? \"#51C1FF\",\r\n      strokeWidth: 0.27,\r\n      fill: \"white\",\r\n    });\r\n\r\n    const topWidth = width / 1.05;\r\n    const topHeight = (15 / 100) * height;\r\n    const topX = x + (width - topWidth) / 2;\r\n    const topY = y + this.verticalMargin;\r\n\r\n    const topContainer = new fabric.Rect({\r\n      left: topX,\r\n      top: topY,\r\n      width: topWidth,\r\n      height: topHeight,\r\n      rx: radius,\r\n      ry: radius,\r\n      stroke: fixtureColor?.primary ?? \"#51C1FF\",\r\n      strokeWidth: 0.27,\r\n      fill: \"white\",\r\n    });\r\n\r\n    const innerBoxHeight = topHeight / 2;\r\n    const innerTopBox = new fabric.Path(\r\n      `M ${topX + radius} ${topY} \r\n              L ${topX + topWidth - radius} ${topY} \r\n              Q ${topX + topWidth} ${topY} ${topX + topWidth} ${topY + radius} \r\n              L ${topX + topWidth} ${topY + innerBoxHeight} \r\n              L ${topX} ${topY + innerBoxHeight} \r\n              L ${topX} ${topY + radius} \r\n              Q ${topX} ${topY} ${topX + radius} ${topY} Z`,\r\n      {\r\n        stroke: fixtureColor?.primary ?? \"#51C1FF\",\r\n        strokeWidth: 0.27,\r\n        fill: \"transparent\",\r\n      }\r\n    );\r\n\r\n    const topText = new fabric.Textbox(\r\n      `Fixture ${fixtureData.associatedElementFixtureNumber\r\n        ? fixtureData.associatedElementFixtureNumber +\r\n        (fixtureData?.fixtureNumber\r\n          ? \" - FX-\" + fixtureData.fixtureNumber\r\n          : \"\")\r\n        : \"new\"\r\n      }`,\r\n      {\r\n        left: topX + topWidth / 2,\r\n        top: topY + innerBoxHeight / 2,\r\n        fontSize: 5,\r\n        fontFamily: \"Inter\",\r\n        fontWeight: \"500\",\r\n        fill: \"#101828\",\r\n        originX: \"center\",\r\n        originY: \"center\",\r\n        width: topWidth,\r\n        height: innerBoxHeight,\r\n        textAlign: \"center\",\r\n      }\r\n    );\r\n\r\n    const secondBoxY = topY + innerBoxHeight;\r\n\r\n    const innerBottomBox = new fabric.Path(\r\n      `M ${topX} ${secondBoxY} \r\n          L ${topX + topWidth} ${secondBoxY} \r\n          L ${topX + topWidth} ${secondBoxY + innerBoxHeight - radius} \r\n          Q ${topX + topWidth} ${secondBoxY + innerBoxHeight} ${topX + topWidth - radius\r\n      } ${secondBoxY + innerBoxHeight} \r\n          L ${topX + radius} ${secondBoxY + innerBoxHeight} \r\n          Q ${topX} ${secondBoxY + innerBoxHeight} ${topX} ${secondBoxY + innerBoxHeight - radius\r\n      } \r\n          L ${topX} ${secondBoxY} Z`,\r\n      {\r\n        fill: \"#344054\",\r\n        stroke: fixtureColor?.primary ?? \"#51C1FF\",\r\n        strokeWidth: 0.27,\r\n      }\r\n    );\r\n\r\n    let fontSize = 5;\r\n    let bottomText;\r\n\r\n    const textContent = `${fixtureData?.fixtureName} - ${fixtureData?.fixtureWidth.value} ${fixtureData?.fixtureWidth.unit}`;\r\n\r\n    const createBottomText = (fs: any) =>\r\n      new fabric.Textbox(textContent, {\r\n        left: topX + topWidth / 2,\r\n        top: secondBoxY + innerBoxHeight / 2,\r\n        fontSize: fs,\r\n        fontFamily: \"Inter\",\r\n        fontWeight: \"500\",\r\n        fill: \"#FFFFFF\",\r\n        originX: \"center\",\r\n        originY: \"center\",\r\n        textAlign: \"center\",\r\n        width: topWidth,\r\n      });\r\n\r\n    do {\r\n      bottomText = createBottomText(fontSize);\r\n      fontSize -= 0.5;\r\n    } while (bottomText.height > innerBoxHeight && fontSize > 2);\r\n\r\n    const midWidth = width / 1.05;\r\n    const midHeight = (77 / 100) * height;\r\n    const midX = x + (width - midWidth) / 2;\r\n    const midY = topY + topHeight + this.verticalMargin;\r\n\r\n    const midContainer = new fabric.Rect({\r\n      left: midX,\r\n      top: midY,\r\n      width: midWidth,\r\n      height: midHeight,\r\n      rx: radius,\r\n      ry: radius,\r\n      stroke: fixtureColor?.primary ?? \"#51C1FF\",\r\n      strokeWidth: 0.27,\r\n      fill: \"white\",\r\n    });\r\n\r\n    const shelfCount = fixtureData.shelfConfig?.length;\r\n    const boxHeight = midHeight / (shelfCount + 2);\r\n    const firstBoxY = midY;\r\n\r\n    const firstBox = new fabric.Path(\r\n      `M ${midX + radius} ${firstBoxY} \r\n              L ${midX + midWidth - radius} ${firstBoxY} \r\n              Q ${midX + midWidth} ${firstBoxY} ${midX + midWidth} ${firstBoxY + radius\r\n      } \r\n              L ${midX + midWidth} ${firstBoxY + boxHeight} \r\n              L ${midX} ${firstBoxY + boxHeight} \r\n              L ${midX} ${firstBoxY + radius} \r\n              Q ${midX} ${firstBoxY} ${midX + radius} ${firstBoxY} Z`,\r\n      {\r\n        fill: fixtureColor?.fill\r\n          ? fixtureColor?.fill\r\n          : fixtureData.issue\r\n            ? \"#D92D20\"\r\n            : \"#EAF8FF\",\r\n        stroke: fixtureColor?.primary\r\n          ? fixtureColor?.primary\r\n          : fixtureData.issue\r\n            ? \"#D92D20\"\r\n            : \"#6BCAFF\",\r\n        strokeWidth: 0.27,\r\n      }\r\n    );\r\n\r\n    const firstBoxText = new fabric.Textbox(\r\n      `${fixtureData?.header?.label ? fixtureData.header.label : \"\"}`,\r\n      {\r\n        left: midX + midWidth / 2,\r\n        top: firstBoxY + boxHeight / 2,\r\n        fontSize: 7,\r\n        fontFamily: \"Inter\",\r\n        fontWeight: \"600\",\r\n        fill: fixtureData.issue ? \"#fff\" : \"#101828\",\r\n        originX: \"center\",\r\n        originY: \"center\",\r\n        width: midWidth,\r\n        height: boxHeight,\r\n        textAlign: \"center\",\r\n      }\r\n    );\r\n\r\n    const shelves = [];\r\n    const shelfHeight = midHeight - 2 * boxHeight;\r\n\r\n    if (fixtureData.productResolutionLevel === \"L1\") {\r\n      const boxY = firstBoxY + boxHeight;\r\n\r\n      const shelf = new fabric.Rect({\r\n        left: midX,\r\n        top: boxY,\r\n        width: midWidth,\r\n        height: shelfHeight,\r\n        // stroke: '#51C1FF',\r\n        stroke: fixtureColor?.primary\r\n          ? fixtureColor?.primary\r\n          : fixtureData.issue\r\n            ? \"#D92D20\"\r\n            : \"#51C1FF\",\r\n        strokeWidth: 0.27,\r\n        fill: \"white\",\r\n      });\r\n\r\n      const text = fixtureData?.productBrandName?.join(\" + \") || \"\";\r\n\r\n      // Wrap text within fixture width and shrink to fit vertical space\r\n      const maxTextWidth = midWidth * 0.9;\r\n      const maxTextHeight = shelfHeight * 0.8;\r\n      let fontSizeL1 = 6;\r\n\r\n      const createWrappedTextbox = (fs: number) =>\r\n        new fabric.Textbox(text, {\r\n          left: midX + midWidth / 2,\r\n          top: boxY + shelfHeight / 2,\r\n          originX: \"center\",\r\n          originY: \"center\",\r\n          width: maxTextWidth,\r\n          fontSize: fs,\r\n          fontFamily: \"Inter\",\r\n          fontWeight: \"500\",\r\n          textAlign: \"center\",\r\n          fill: \"#333\",\r\n          selectable: false,\r\n          evented: false,\r\n        });\r\n\r\n      let wrappedText = createWrappedTextbox(fontSizeL1);\r\n      while (wrappedText.height > maxTextHeight && fontSizeL1 > 2) {\r\n        fontSizeL1 -= 0.5;\r\n        wrappedText = createWrappedTextbox(fontSizeL1);\r\n      }\r\n\r\n      const padding = 2;\r\n      const background = new fabric.Rect({\r\n        left: midX + midWidth / 2,\r\n        top: boxY + shelfHeight / 2,\r\n        originX: \"center\",\r\n        originY: \"center\",\r\n        width: Math.min(maxTextWidth, wrappedText.width!) + padding,\r\n        height: wrappedText.getScaledHeight() + 2,\r\n        fill: \"rgba(255, 255, 255, 0.5)\",\r\n        selectable: false,\r\n        evented: false,\r\n        rx: 1,\r\n        ry: 1,\r\n      });\r\n\r\n      shelves.push(shelf, background, wrappedText);\r\n    } else if (fixtureData.productResolutionLevel === \"L2\") {\r\n      for (let i = 0; i < shelfCount; i++) {\r\n        const boxY = firstBoxY + boxHeight * (i + 1);\r\n\r\n        const shelf = new fabric.Rect({\r\n          left: midX,\r\n          top: boxY,\r\n          width: midWidth,\r\n          height: boxHeight,\r\n          stroke: fixtureColor?.primary\r\n            ? fixtureColor?.primary\r\n            : fixtureData.issue\r\n              ? \"#D92D20\"\r\n              : \"#51C1FF\",\r\n          strokeWidth: 0.27,\r\n          fill: \"white\",\r\n        });\r\n\r\n        shelves.push(shelf);\r\n\r\n        if (fixtureData.shelfConfig?.[i].shelfType === \"tray\") {\r\n          const trayRows = fixtureData.shelfConfig?.[i].trayRows || 1;\r\n          const rowSpacing = boxHeight / trayRows;\r\n\r\n          for (let r = 1; r < trayRows; r++) {\r\n            const lineY = boxY + r * rowSpacing;\r\n\r\n            const trayLine = new fabric.Line(\r\n              [midX, lineY, midX + midWidth, lineY],\r\n              {\r\n                stroke: \"#999\",\r\n                strokeWidth: 0.2,\r\n                selectable: false,\r\n                evented: false,\r\n              }\r\n            );\r\n\r\n            shelves.push(trayLine);\r\n          }\r\n        }\r\n\r\n        const text =\r\n          fixtureData.shelfConfig?.[i].productBrandName?.join(\" + \") || \"\";\r\n\r\n        // L2: Single-line ellipsis against full shelf width\r\n        const horizontalPadding = 6;\r\n        const maxTextWidth = Math.max(0, midWidth - horizontalPadding);\r\n        const maxTextHeight = boxHeight * 0.8;\r\n        const baseFontSize = 5;\r\n\r\n        const measure = (t: string) =>\r\n          new fabric.Text(t, {\r\n            fontSize: baseFontSize,\r\n            fontFamily: \"Inter\",\r\n            fontWeight: \"500\",\r\n          });\r\n\r\n        let displayText = text;\r\n        let measured = measure(displayText);\r\n\r\n        if ((measured.width ?? 0) > maxTextWidth) {\r\n          const ellipsis = \"...\";\r\n          let leftIdx = 0;\r\n          let rightIdx = text.length;\r\n\r\n          while (leftIdx < rightIdx) {\r\n            const midIdx = Math.floor((leftIdx + rightIdx) / 2);\r\n            const candidate = text.slice(0, midIdx) + ellipsis;\r\n            measured = measure(candidate);\r\n            if ((measured.width ?? 0) > maxTextWidth) {\r\n              rightIdx = midIdx;\r\n            } else {\r\n              leftIdx = midIdx + 1;\r\n            }\r\n          }\r\n          displayText = text.slice(0, Math.max(0, leftIdx - 1)) + ellipsis;\r\n          measured = measure(displayText);\r\n        }\r\n\r\n        const background = new fabric.Rect({\r\n          left: midX + midWidth / 2,\r\n          top: boxY + boxHeight / 2,\r\n          originX: \"center\",\r\n          originY: \"center\",\r\n          width: midWidth,\r\n          height: Math.min(maxTextHeight, measured.height ?? 10) + 2,\r\n          fill: \"rgba(255, 255, 255, 0.5)\",\r\n          selectable: false,\r\n          evented: false,\r\n          rx: 1,\r\n          ry: 1,\r\n        });\r\n\r\n        const finalText = new fabric.Text(displayText, {\r\n          left: midX + midWidth / 2,\r\n          top: boxY + boxHeight / 2,\r\n          originX: \"center\",\r\n          originY: \"center\",\r\n          fontSize: baseFontSize,\r\n          fontFamily: \"Inter\",\r\n          fontWeight: \"500\",\r\n          fill: \"#333\",\r\n          selectable: false,\r\n          evented: false,\r\n        });\r\n\r\n        shelves.push(background, finalText);\r\n      }\r\n    } else if (fixtureData.productResolutionLevel === \"L3\") {\r\n      const sequentialGroups: { zone: string; shelves: any[] }[] = [];\r\n\r\n      let previousZone: string | null = null;\r\n\r\n      for (const shelf of fixtureData.shelfConfig) {\r\n        const zone = shelf.zone || \"Unknown\";\r\n        if (zone !== previousZone) {\r\n          sequentialGroups.push({ zone, shelves: [shelf] });\r\n          previousZone = zone;\r\n        } else {\r\n          sequentialGroups[sequentialGroups.length - 1].shelves.push(shelf);\r\n        }\r\n      }\r\n\r\n      let currentShelfIndex = 0;\r\n\r\n      for (const group of sequentialGroups) {\r\n        const { zone, shelves: shelvesInGroup } = group;\r\n\r\n        const groupStartIndex = currentShelfIndex;\r\n        const groupLength = shelvesInGroup.length;\r\n\r\n        const groupTopY = firstBoxY + boxHeight * (groupStartIndex + 1);\r\n        const groupBottomY = groupTopY + boxHeight * groupLength;\r\n\r\n        for (let j = 0; j < shelvesInGroup.length; j++) {\r\n          const shelfData = shelvesInGroup[j];\r\n          const boxY = firstBoxY + boxHeight * (currentShelfIndex + 1);\r\n\r\n          const shelfRect = new fabric.Rect({\r\n            left: midX,\r\n            top: boxY,\r\n            width: midWidth,\r\n            height: boxHeight,\r\n            stroke: fixtureColor?.primary\r\n              ? fixtureColor?.primary\r\n              : fixtureData.issue\r\n                ? \"#D92D20\"\r\n                : \"#51C1FF\",\r\n            strokeWidth: 0.27,\r\n            fill: \"white\",\r\n          });\r\n\r\n          shelves.push(shelfRect);\r\n\r\n          if (shelfData.shelfType === \"tray\") {\r\n            const trayRows = shelfData.trayRows || 0;\r\n            const productPerShelf = shelfData.productPerShelf || 0;\r\n\r\n            const rowSpacing = boxHeight / trayRows;\r\n            const columnSpacing = midWidth / productPerShelf;\r\n\r\n            for (let r = 1; r < trayRows; r++) {\r\n              const lineY = boxY + r * rowSpacing;\r\n\r\n              const trayLine = new fabric.Line(\r\n                [midX, lineY, midX + midWidth, lineY],\r\n                {\r\n                  stroke: \"#999\",\r\n                  strokeWidth: 0.2,\r\n                  selectable: false,\r\n                  evented: false,\r\n                }\r\n              );\r\n\r\n              shelves.push(trayLine);\r\n            }\r\n\r\n            for (let c = 1; c < productPerShelf; c++) {\r\n              const lineX = midX + c * columnSpacing;\r\n\r\n              const verticalLine = new fabric.Line(\r\n                [lineX, boxY, lineX, boxY + boxHeight],\r\n                {\r\n                  stroke: \"#999\",\r\n                  strokeWidth: 0.2,\r\n                  selectable: false,\r\n                  evented: false,\r\n                }\r\n              );\r\n\r\n              shelves.push(verticalLine);\r\n            }\r\n          }\r\n\r\n          currentShelfIndex++;\r\n        }\r\n\r\n        const groupBottomLine = new fabric.Line(\r\n          [midX, groupBottomY, midX + midWidth, groupBottomY],\r\n          {\r\n            stroke: fixtureColor?.primary\r\n              ? fixtureColor?.primary\r\n              : fixtureData.issue\r\n                ? \"#D92D20\"\r\n                : \"#51C1FF\",\r\n            strokeWidth: 1,\r\n            selectable: false,\r\n            evented: false,\r\n          }\r\n        );\r\n\r\n        shelves.push(groupBottomLine);\r\n\r\n        if (fixtureData.fixtureType === \"wall\") {\r\n          const labelPadding = 2;\r\n          const labelText = new fabric.Textbox(zone, {\r\n            left: midX + midWidth - labelPadding,\r\n            top: groupBottomY,\r\n            fontSize: 5,\r\n            fontFamily: \"Inter\",\r\n            fontWeight: \"600\",\r\n            fill: \"#707070\",\r\n            originX: \"right\",\r\n            originY: \"bottom\",\r\n            width: midWidth / 2,\r\n            textAlign: \"right\",\r\n          });\r\n\r\n          shelves.push(labelText);\r\n        }\r\n\r\n        const groupCenterX = midX + midWidth / 2;\r\n        const groupCenterY = groupTopY + (groupBottomY - groupTopY) / 2;\r\n\r\n        const sectionBrands = new Set();\r\n\r\n        shelvesInGroup?.forEach((shelf) => {\r\n          shelf.productBrandName?.forEach((brand: string) => {\r\n            sectionBrands.add(brand);\r\n          });\r\n        });\r\n\r\n        if ([...sectionBrands].length) {\r\n          const text = [...sectionBrands].join(\" + \");\r\n          const padding = 2;\r\n          const maxTextWidth = midWidth;\r\n          const rectWidth = midWidth * 0.9;\r\n          const maxTextHeight = 20;\r\n\r\n          let fontSize = 6;\r\n          let wrappedText;\r\n\r\n          const createTextbox = (fs: any) => {\r\n            return new fabric.Textbox(text, {\r\n              fontSize: fs,\r\n              fontFamily: \"Inter\",\r\n              fontWeight: \"400\",\r\n              width: maxTextWidth,\r\n              textAlign: \"center\",\r\n              originX: \"center\",\r\n              originY: \"center\",\r\n              fill: \"#101828\",\r\n              selectable: false,\r\n              evented: false,\r\n              left: groupCenterX,\r\n              top: groupCenterY,\r\n            });\r\n          };\r\n\r\n          do {\r\n            wrappedText = createTextbox(fontSize);\r\n            fontSize -= 0.5;\r\n          } while (wrappedText.height > maxTextHeight && fontSize > 2);\r\n\r\n          const rectHeight = wrappedText.getScaledHeight() + padding;\r\n\r\n          const groupCenterText = new fabric.Rect({\r\n            left: groupCenterX,\r\n            top: groupCenterY,\r\n            originX: \"center\",\r\n            originY: \"center\",\r\n            width: rectWidth,\r\n            height: rectHeight,\r\n            fill: \"rgba(255, 255, 255, 0.5)\",\r\n            selectable: false,\r\n            evented: false,\r\n            rx: 1,\r\n            ry: 1,\r\n          });\r\n\r\n          shelves.push(groupCenterText, wrappedText);\r\n        }\r\n      }\r\n    }\r\n\r\n    if (fixtureData?.vmConfig?.length) {\r\n      for (let i = 0; i < fixtureData.vmConfig.length; i++) {\r\n        const vm = fixtureData.vmConfig[i];\r\n\r\n        const startIndex = vm.startYPosition - 1;\r\n        const endIndex = vm.endYPosition - 1;\r\n\r\n        const shelfTopY = firstBoxY + boxHeight * (startIndex + 1);\r\n        const shelfHeightSpan = boxHeight * (endIndex - startIndex + 1);\r\n\r\n        let vmHeightPx: number = shelfHeightSpan;\r\n        let y: number = shelfTopY;\r\n\r\n        if (vm.yZone && vm.yZone !== \"stretch\") {\r\n          vmHeightPx = shelfHeightSpan / 3;\r\n\r\n          switch (vm.yZone) {\r\n            case \"top\":\r\n              y = shelfTopY;\r\n              break;\r\n            case \"middle\":\r\n              y = shelfTopY + vmHeightPx;\r\n              break;\r\n            case \"bottom\":\r\n              y = shelfTopY + 2 * vmHeightPx;\r\n              break;\r\n            default:\r\n              y = shelfTopY;\r\n          }\r\n        }\r\n\r\n        let x = midX;\r\n        let width = midWidth;\r\n\r\n        if (!vm.xZone || vm.xZone === \"stretch\") {\r\n          width = midWidth;\r\n          x = midX;\r\n        } else {\r\n          const sectionWidth = midWidth / 3;\r\n          switch (vm.xZone) {\r\n            case \"left\":\r\n              width = sectionWidth;\r\n              x = midX;\r\n              break;\r\n            case \"middle\":\r\n              width = sectionWidth;\r\n              x = midX + sectionWidth;\r\n              break;\r\n            case \"right\":\r\n              width = sectionWidth;\r\n              x = midX + 2 * sectionWidth;\r\n              break;\r\n            default:\r\n              width = midWidth;\r\n              x = midX;\r\n          }\r\n        }\r\n\r\n        if (vm.vmImageUrl) {\r\n          try {\r\n            const vmImage: any = await fabric.FabricImage.fromURL(\r\n              `${this.cdnUrl}${vm.vmImageUrl}?v=${Date.now()}`,\r\n              {\r\n                crossOrigin: \"anonymous\",\r\n              }\r\n            );\r\n\r\n            vmImage.set({\r\n              left: x + width / 2,\r\n              top: y + vmHeightPx / 2,\r\n              originX: \"center\",\r\n              originY: \"center\",\r\n              scaleX: width / vmImage.width!,\r\n              scaleY: vmHeightPx / vmImage.height!,\r\n              selectable: false,\r\n              evented: false,\r\n            });\r\n\r\n            shelves.push(vmImage);\r\n          } catch (err) {\r\n            console.error(`Error loading image for VM: ${vm.vmImageUrl}`, err);\r\n          }\r\n        } else {\r\n          const vmBox = new fabric.Rect({\r\n            left: x,\r\n            top: y,\r\n            width,\r\n            height: vmHeightPx,\r\n            fill: fixtureColor?.vmFill ?? \"#6938EF\",\r\n            selectable: false,\r\n            evented: false,\r\n          });\r\n\r\n          let fontSize = 5;\r\n          let vmText;\r\n\r\n          const createVmText = (fs: number) =>\r\n            new fabric.Textbox(vm.vmName || \"\", {\r\n              left: x + width / 2,\r\n              top: y + vmHeightPx / 2,\r\n              originX: \"center\",\r\n              originY: \"center\",\r\n              fontSize: fs,\r\n              fontFamily: \"Inter\",\r\n              fontWeight: \"600\",\r\n              fill: \"white\",\r\n              textAlign: \"center\",\r\n              width: width,\r\n              selectable: false,\r\n              evented: false,\r\n            });\r\n\r\n          do {\r\n            vmText = createVmText(fontSize);\r\n            fontSize -= 0.5;\r\n          } while (vmText.height > vmHeightPx && fontSize > 2);\r\n\r\n          shelves.push(vmBox, vmText);\r\n        }\r\n      }\r\n    }\r\n\r\n    const lastBoxY = firstBoxY + boxHeight * (shelfCount + 1);\r\n    const lastBox = new fabric.Path(\r\n      `M ${midX} ${lastBoxY} \r\n              L ${midX + midWidth} ${lastBoxY} \r\n              L ${midX + midWidth} ${lastBoxY + boxHeight - radius} \r\n              Q ${midX + midWidth} ${lastBoxY + boxHeight} ${midX + midWidth - radius\r\n      } ${lastBoxY + boxHeight} \r\n              L ${midX + radius} ${lastBoxY + boxHeight} \r\n              Q ${midX} ${lastBoxY + boxHeight} ${midX} ${lastBoxY + boxHeight - radius\r\n      } \r\n              L ${midX} ${lastBoxY} Z`,\r\n      {\r\n        fill: fixtureColor?.fill\r\n          ? fixtureColor?.fill\r\n          : fixtureData.issue\r\n            ? \"#FEE4E2\"\r\n            : \"#EAF8FF\",\r\n        stroke: fixtureColor?.primary\r\n          ? fixtureColor?.primary\r\n          : fixtureData.issue\r\n            ? \"#D92D20\"\r\n            : \"#6BCAFF\",\r\n        strokeWidth: 0.27,\r\n      }\r\n    );\r\n\r\n    const lastBoxText = new fabric.Textbox(\r\n      `${fixtureData?.footer?.label ? fixtureData.footer.label : \"\"}`,\r\n      {\r\n        left: midX + midWidth / 2,\r\n        top: lastBoxY + boxHeight / 2,\r\n        fontSize: 5,\r\n        fontFamily: \"Inter\",\r\n        fontWeight: \"600\",\r\n        fill: \"#101828\",\r\n        originX: \"center\",\r\n        originY: \"center\",\r\n        textAlign: \"center\",\r\n        width: midWidth,\r\n        heigth: boxHeight,\r\n      }\r\n    );\r\n\r\n    const bottomWidth = width / 1.05;\r\n    const bottomHeight = (3 / 100) * height;\r\n    const bottomX = x + (width - bottomWidth) / 2;\r\n    const bottomY = midY + midHeight + this.verticalMargin;\r\n\r\n    const bottomContainer = new fabric.Rect({\r\n      left: bottomX,\r\n      top: bottomY,\r\n      width: bottomWidth,\r\n      height: bottomHeight,\r\n      rx: radius,\r\n      ry: radius,\r\n      stroke: \"transparent\",\r\n      fill: \"transparent\",\r\n    });\r\n\r\n    const bottomTextContent = new fabric.Textbox(\r\n      `Shelves - ${fixtureData.shelfConfig?.length} . Products -  ${fixtureData.fixtureCapacity} . VM - ${fixtureData.vmConfig?.length}`,\r\n      {\r\n        left: bottomX + bottomWidth / 2,\r\n        top: bottomY + bottomHeight / 2,\r\n        fontSize: 5,\r\n        fontFamily: \"Inter\",\r\n        fontWeight: \"500\",\r\n        fill: \"#333\",\r\n        originX: \"center\",\r\n        originY: \"center\",\r\n        textAlign: \"center\",\r\n        width: bottomWidth,\r\n        height: bottomHeight,\r\n      }\r\n    );\r\n\r\n\r\n    const fixtureGroup = new fabric.Group(\r\n      [\r\n        mainContainer,\r\n        topContainer,\r\n        innerTopBox,\r\n        topText,\r\n        innerBottomBox,\r\n        bottomText,\r\n        midContainer,\r\n        firstBox,\r\n        firstBoxText,\r\n        ...shelves,\r\n        lastBox,\r\n        lastBoxText,\r\n        bottomContainer,\r\n        bottomTextContent,\r\n      ],\r\n      {\r\n        left: x,\r\n        top: y,\r\n        evented: true,\r\n        hoverCursor: \"pointer\",\r\n        lockScalingX: true,\r\n        lockScalingY: true,\r\n        subTargetCheck: true,\r\n        // hasControls: fixtureData.fixtureType === \"floor\",\r\n        // lockRotation: fixtureData.fixtureType !== \"floor\",\r\n        // lockMovementX: currentFixtureInfo?.wallDirection === 'up' || currentFixtureInfo?.wallDirection === 'down',\r\n        // lockMovementY: currentFixtureInfo?.wallDirection === 'right' || currentFixtureInfo?.wallDirection === 'left',\r\n        angle: fixtureData.relativePosition?.angle ?? 0,\r\n      }\r\n    ) as fabric.Group & {\r\n      wallIndex: number;\r\n      fixtureIndex: number;\r\n      fixtureId: string;\r\n      fixtureType: \"wall\" | \"floor\";\r\n      issue: boolean;\r\n      issueData: any;\r\n      taskType: string;\r\n      vmConfig: [];\r\n      objType: string;\r\n    };\r\n\r\n\r\n\r\n    fixtureGroup.fixtureType = fixtureData.fixtureType;\r\n    fixtureGroup.wallIndex = fixtureData.associatedElementNumber;\r\n    fixtureGroup.fixtureIndex = fixtureData.associatedElementFixtureNumber;\r\n    fixtureGroup.fixtureId = fixtureData._id;\r\n    fixtureGroup.issue = fixtureData.issue;\r\n    fixtureGroup.issueData = fixtureData.issueData;\r\n    fixtureGroup.taskType = fixtureData.taskType;\r\n    fixtureGroup.vmConfig = fixtureData.vmConfig;\r\n    fixtureGroup.objType = \"fixture\";\r\n\r\n\r\n    canvas.add(fixtureGroup);\r\n\r\n    // If newly added, focus it once\r\n    if ((fixtureData as any)?.newAdded) {\r\n      try {\r\n        canvas.setActiveObject(fixtureGroup);\r\n        canvas.requestRenderAll();\r\n      } catch { }\r\n      delete (fixtureData as any).newAdded;\r\n    }\r\n\r\n    return fixtureGroup;\r\n  }\r\n\r\n  drawOtherElementsWithLabel(\r\n    canvas: fabric.Canvas,\r\n    x: number,\r\n    y: number,\r\n    a: number,\r\n    width: number,\r\n    height: number,\r\n    element: any\r\n  ): fabric.Group {\r\n    const rect = new fabric.Rect({\r\n      width,\r\n      height,\r\n      rx: 2,\r\n      ry: 2,\r\n      fill: \"#DAF1FF\",\r\n      stroke: \"#6BCAFF\",\r\n      strokeWidth: 2,\r\n      originX: \"center\",\r\n      originY: \"center\",\r\n    });\r\n\r\n    const text = new fabric.Textbox(element.fixtureName, {\r\n      width: width - 8,\r\n      fontSize: 14,\r\n      fontFamily: \"Inter\",\r\n      fontWeight: \"500\",\r\n      fill: \"#101828\",\r\n      originX: \"center\",\r\n      originY: \"center\",\r\n      textAlign: \"center\",\r\n      editable: false,\r\n      splitByGrapheme: true,\r\n    });\r\n\r\n    const textBounds = text.getBoundingRect();\r\n\r\n    if (textBounds.height > height - 8) {\r\n      const scale = (height - 8) / textBounds.height;\r\n      text.set(\"fontSize\", text.fontSize! * scale);\r\n    }\r\n\r\n    const group = new fabric.Group([rect, text], {\r\n      left: x,\r\n      top: y,\r\n      angle: a,\r\n      originX: \"center\",\r\n      originY: \"center\",\r\n    }) as fabric.Group & {\r\n      objType: string;\r\n      data: any;\r\n    };\r\n\r\n    group.objType = \"others\";\r\n    group.data = element;\r\n\r\n    canvas.add(group);\r\n\r\n    if ((element as any)?.newAdded) {\r\n      try {\r\n        canvas.setActiveObject(group);\r\n        canvas.requestRenderAll();\r\n      } catch { }\r\n      delete (element as any).newAdded;\r\n    }\r\n\r\n    return group;\r\n  }\r\n\r\n\r\n  highlightFixture(canvas: fabric.Canvas, group: any, blink: boolean = false) {\r\n    this.removeHighlight(canvas);\r\n    this.cancelHighlightBlink = false;\r\n\r\n    const center = group.getCenterPoint();\r\n\r\n    const realWidth = group.width! * group.scaleX!;\r\n    const realHeight = group.height! * group.scaleY!;\r\n\r\n    const rect = new fabric.Rect({\r\n      left: center.x,\r\n      top: center.y,\r\n      width: realWidth + 8, // consistent padding\r\n      height: realHeight + 8,\r\n      rx: 5,\r\n      ry: 5,\r\n      stroke: \"#00aaff\",\r\n      strokeWidth: 3,\r\n      fill: \"transparent\",\r\n      selectable: false,\r\n      evented: false,\r\n      originX: \"center\",\r\n      originY: \"center\",\r\n      angle: group.angle || 0,\r\n      opacity: blink ? 0 : 1,\r\n    });\r\n\r\n    (rect as any).data = { type: \"fixture-highlight\" };\r\n\r\n    canvas.add(rect);\r\n    canvas.moveObjectTo(rect, canvas.getObjects().length - 1);\r\n    canvas.requestRenderAll();\r\n\r\n    this.activeHighlight = rect;\r\n\r\n    // --- No blinking needed ---\r\n    if (!blink) return;\r\n\r\n    let cycles = 0;\r\n\r\n    const blinkOnce = () => {\r\n      if (this.cancelHighlightBlink) {\r\n        this.removeHighlight(canvas);\r\n        return;\r\n      }\r\n\r\n      rect.animate(\r\n        { opacity: 1 },\r\n        {\r\n          duration: 200,\r\n          onChange: () => canvas.requestRenderAll(),\r\n          onComplete: () => {\r\n            if (this.cancelHighlightBlink) {\r\n              this.removeHighlight(canvas);\r\n              return;\r\n            }\r\n\r\n            rect.animate(\r\n              { opacity: 0 },\r\n              {\r\n                duration: 200,\r\n                onChange: () => canvas.requestRenderAll(),\r\n                onComplete: () => {\r\n                  if (this.cancelHighlightBlink) {\r\n                    this.removeHighlight(canvas);\r\n                    return;\r\n                  }\r\n\r\n                  cycles++;\r\n                  if (cycles < 20) {\r\n                    blinkOnce();\r\n                  } else {\r\n                    if (!this.cancelHighlightBlink) {\r\n                      rect.set({ opacity: 1 });\r\n                      canvas.requestRenderAll();\r\n                    }\r\n                  }\r\n                },\r\n              }\r\n            );\r\n          },\r\n        }\r\n      );\r\n    };\r\n\r\n    blinkOnce();\r\n  }\r\n\r\n  removeHighlight(canvas: fabric.Canvas,) {\r\n    this.cancelHighlightBlink = true;\r\n\r\n    canvas.getObjects().forEach((obj) => {\r\n      if ((obj as any).data?.type === \"fixture-highlight\") {\r\n        canvas.remove(obj);\r\n      }\r\n    });\r\n\r\n    this.activeHighlight = null;\r\n    canvas.requestRenderAll();\r\n  }\r\n\r\n  async applyStores() {\r\n    const value = this.form.value.stores;\r\n    if (!value) return;\r\n\r\n    const codes = value\r\n      .split(',')\r\n      .map((s: string) => s.trim().toUpperCase())\r\n      .filter(Boolean);\r\n\r\n    const response: any = await lastValueFrom(\r\n      this.apiService.getPlanoId({ stores: codes })\r\n    );\r\n\r\n    const data = Array.isArray(response.data) ? response.data : [];\r\n\r\n    const foundStoreSet = new Set(\r\n      data.map((store: any) => store.storeName.toUpperCase())\r\n    );\r\n\r\n    const foundChips = data.map((store: any) => ({\r\n      storeCode: store.storeName,\r\n      status: 'pending',\r\n      planoId: store._id\r\n    }));\r\n\r\n    const notFoundChips = codes\r\n      .filter((code: string) => !foundStoreSet.has(code))\r\n      .map((code: string) => ({\r\n        storeCode: code,\r\n        status: 'not_found'\r\n      }));\r\n\r\n    this.storeChips = [...foundChips, ...notFoundChips];\r\n    const activeChip = this.storeChips.find((chip) => chip.status === 'pending')\r\n    this.selectedStore = activeChip?.storeCode\r\n    this.getStoreFixtures(activeChip!.planoId)\r\n  }\r\n\r\n  private delay(ms: number): Promise<void> {\r\n    return new Promise(resolve => setTimeout(resolve, ms));\r\n  }\r\n\r\n  async autoApprove() {\r\n    for (let i = 0; i < this.storeChips.length; i++) {\r\n      const { storeCode, status, planoId } = this.storeChips[i];\r\n      if (status === 'not_found') continue;\r\n\r\n      const planoRes: any = await lastValueFrom(this.apiService.getStoreFixtures({ id: [planoId] }))\r\n\r\n      if (!planoRes?.data?.[0]?.floors?.length) {\r\n        return;\r\n      }\r\n\r\n      this.planoData = structuredClone(planoRes.data[0]);\r\n\r\n      this.floorData = structuredClone(planoRes.data[0].floors[0]);\r\n\r\n      this.renderFloor(this.beforeCanvas);\r\n      this.renderFloor(this.afterCanvas);\r\n\r\n      this.loading = false;\r\n      setTimeout(() => {\r\n        this.resizeCanvas(this.beforeCanvas, this.beforeContainerRef);\r\n        this.fitCanvasToLayoutAnimated(this.beforeCanvas);\r\n\r\n        this.resizeCanvas(this.afterCanvas, this.afterContainerRef);\r\n        this.fitCanvasToLayoutAnimated(this.afterCanvas);\r\n      }, 100);\r\n\r\n      const payload = {\r\n        ...this.form.value,\r\n        stores: [storeCode]\r\n      };\r\n\r\n      const aiRes: any = await lastValueFrom(this.apiService.findFixtureAi(payload))\r\n\r\n      if (aiRes.code === 200) {\r\n        this.updateAiResponseFixture(aiRes.data)\r\n        this.renderFloor(this.afterCanvas)\r\n\r\n\r\n        setTimeout(() => {\r\n          const actionFixture = this.afterCanvas\r\n            .getObjects()\r\n            .find(\r\n              (obj: any) => {\r\n                return obj.objType === \"fixture\" && obj.fixtureId === aiRes.data._id\r\n              }\r\n            ) as fabric.Group | undefined;\r\n\r\n          if (!actionFixture) return;\r\n          this.highlightFixture(this.afterCanvas, actionFixture, true);\r\n        }, 0);\r\n      }\r\n\r\n      const updatePayload = {\r\n        fixtures: [this.selectedFixture]\r\n      }\r\n\r\n      await lastValueFrom(this.apiService.updateFixtureData(updatePayload))\r\n      this.storeChips[i].status = 'approved'\r\n      const activeChip = this.storeChips.find((chip) => chip.status === 'pending')\r\n      this.selectedStore = activeChip?.storeCode\r\n\r\n      await this.delay(3000);\r\n\r\n    }\r\n  }\r\n\r\n  approveStore() {\r\n    const payload = {\r\n      fixtures: [this.selectedFixture]\r\n    }\r\n\r\n    this.apiService.updateFixtureData(payload).subscribe({\r\n      next: (res: any) => {\r\n        if (res.code === 200) {\r\n          this.storeChips.find((chip) => chip.storeCode === this.selectedStore)!.status = 'approved'\r\n          const nextChip = this.storeChips.find((chip) => chip.status === 'pending')\r\n          this.selectedStore = nextChip?.storeCode\r\n          this.getStoreFixtures(nextChip!.planoId)\r\n        }\r\n      }\r\n    })\r\n  }\r\n\r\n  selectStore(chip: StoreChip) {\r\n    this.selectedStore = chip.storeCode\r\n    this.getStoreFixtures(chip.planoId)\r\n  }\r\n}\r\n","<form class=\"ai-container\" [formGroup]=\"form\" (ngSubmit)=\"submitPrompt()\">\r\n\r\n  <!-- ================= Applicable Stores ================= -->\r\n  <div class=\"field\">\r\n    <label>Applicable Stores</label>\r\n\r\n    <div class=\"d-flex gap-2 align-items-center\">\r\n      <input type=\"text\" class=\"form-control\" formControlName=\"stores\" placeholder=\"e.g. LKST345, LKST346\" />\r\n\r\n      <button type=\"button\" class=\"btn btn-outline\" (click)=\"applyStores()\">\r\n        Apply\r\n      </button>\r\n    </div>\r\n\r\n    <span class=\"error\" *ngIf=\"form.controls.stores.touched && form.controls.stores.invalid\">\r\n      Applicable stores is required\r\n    </span>\r\n  </div>\r\n\r\n  <!-- ================= Store Chips ================= -->\r\n  <div class=\"store-chips\" *ngIf=\"storeChips.length\">\r\n    <div (click)=\"selectStore(chip)\" class=\"store-chip\" *ngFor=\"let chip of storeChips\" [ngClass]=\"[\r\n      chip.status,\r\n      chip.storeCode === selectedStore ? 'active' : ''\r\n    ]\">\r\n      {{ chip.storeCode }}\r\n    </div>\r\n  </div>\r\n\r\n\r\n  <!-- ================= Update Type ================= -->\r\n  <div class=\"field\">\r\n    <label>Update Type</label>\r\n\r\n    <div class=\"radio-group\">\r\n      <label class=\"radio-option\">\r\n        <input type=\"radio\" formControlName=\"updateType\" value=\"vm\" />\r\n        VM\r\n      </label>\r\n\r\n      <label class=\"radio-option\">\r\n        <input type=\"radio\" formControlName=\"updateType\" value=\"pid\" />\r\n        PID\r\n      </label>\r\n    </div>\r\n\r\n    <span class=\"error\" *ngIf=\"form.controls.updateType.touched && form.controls.updateType.invalid\">\r\n      Update type is required\r\n    </span>\r\n  </div>\r\n\r\n  <!-- ================= AI Prompt ================= -->\r\n  <div class=\"field\">\r\n    <label>AI Prompt</label>\r\n\r\n    <div class=\"row\">\r\n\r\n      <!-- Selection Criteria -->\r\n      <div class=\"col-md-6\">\r\n        <div class=\"form-group\">\r\n          <label class=\"small fw-semibold\">Selection Criteria</label>\r\n\r\n          <textarea rows=\"3\" class=\"form-control\" formControlName=\"selectionCriteria\"\r\n            placeholder=\"e.g. Select premium shades on wall fixtures\"></textarea>\r\n\r\n          <span class=\"error\"\r\n            *ngIf=\"form.controls.selectionCriteria.touched && form.controls.selectionCriteria.invalid\">\r\n            Selection criteria is required\r\n          </span>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Updation Criteria -->\r\n      <div class=\"col-md-6\">\r\n        <div class=\"form-group\">\r\n          <label class=\"small fw-semibold\">Updation Criteria</label>\r\n\r\n          <textarea rows=\"3\" class=\"form-control\" formControlName=\"updationCriteria\"\r\n            placeholder=\"e.g. Move selected fixtures to mid shelves\"></textarea>\r\n\r\n          <span class=\"error\" *ngIf=\"form.controls.updationCriteria.touched && form.controls.updationCriteria.invalid\">\r\n            Updation criteria is required\r\n          </span>\r\n        </div>\r\n      </div>\r\n\r\n    </div>\r\n  </div>\r\n\r\n  <!-- ================= Submit ================= -->\r\n  <div class=\"d-flex justify-content-end mt-3\">\r\n    <button type=\"submit\" class=\"submit-btn\" [disabled]=\"loading\">\r\n      {{ loading ? 'Thinking…' : 'Submit' }}\r\n    </button>\r\n  </div>\r\n\r\n  <!-- ================= Canvases ================= -->\r\n  <div class=\"canvas-grid mt-4\">\r\n\r\n    <div class=\"canvas-box\" #beforeCanvasContainer>\r\n      <div class=\"canvas-title\">Before Update</div>\r\n      <canvas #beforeCanvas></canvas>\r\n    </div>\r\n\r\n    <div class=\"canvas-box\" #afterCanvasContainer>\r\n      <div class=\"canvas-title\">After Update</div>\r\n      <canvas #afterCanvas></canvas>\r\n    </div>\r\n\r\n  </div>\r\n\r\n  <div class=\"d-flex align-items-center justify-content-end\">\r\n    <!-- ================= Auto Approve ================= -->\r\n    <button (click)=\"autoApprove()\" type=\"button\" class=\"btn btn-warning me-3\">\r\n      ⚠ Auto Approve\r\n    </button>\r\n\r\n    <!-- ================= Approve Changes ================= -->\r\n    <button type=\"button\" class=\"btn btn-success\" (click)=\"approveStore()\" [disabled]=\"form.invalid\">\r\n      ✔ Approve Changes\r\n    </button>\r\n  </div>\r\n\r\n\r\n</form>"]}
|