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.
@@ -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>"]}