homebridge-roborock-vacuum 0.1.0

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.
Files changed (45) hide show
  1. package/CHANGELOG.md +5 -0
  2. package/LICENSE +21 -0
  3. package/README.md +37 -0
  4. package/config.schema.json +31 -0
  5. package/dist/index.js +10 -0
  6. package/dist/index.js.map +1 -0
  7. package/dist/logger.js +39 -0
  8. package/dist/logger.js.map +1 -0
  9. package/dist/platform.js +167 -0
  10. package/dist/platform.js.map +1 -0
  11. package/dist/settings.js +8 -0
  12. package/dist/settings.js.map +1 -0
  13. package/dist/types.js +3 -0
  14. package/dist/types.js.map +1 -0
  15. package/dist/vacuum_accessory.js +152 -0
  16. package/dist/vacuum_accessory.js.map +1 -0
  17. package/package.json +66 -0
  18. package/roborockLib/data/UserData +4 -0
  19. package/roborockLib/data/clientID +4 -0
  20. package/roborockLib/i18n/de/translations.json +188 -0
  21. package/roborockLib/i18n/en/translations.json +208 -0
  22. package/roborockLib/i18n/es/translations.json +188 -0
  23. package/roborockLib/i18n/fr/translations.json +188 -0
  24. package/roborockLib/i18n/it/translations.json +188 -0
  25. package/roborockLib/i18n/nl/translations.json +188 -0
  26. package/roborockLib/i18n/pl/translations.json +188 -0
  27. package/roborockLib/i18n/pt/translations.json +188 -0
  28. package/roborockLib/i18n/ru/translations.json +188 -0
  29. package/roborockLib/i18n/uk/translations.json +188 -0
  30. package/roborockLib/i18n/zh-cn/translations.json +188 -0
  31. package/roborockLib/lib/RRMapParser.js +447 -0
  32. package/roborockLib/lib/deviceFeatures.js +995 -0
  33. package/roborockLib/lib/localConnector.js +249 -0
  34. package/roborockLib/lib/map/map.html +110 -0
  35. package/roborockLib/lib/map/zones.js +713 -0
  36. package/roborockLib/lib/mapCreator.js +692 -0
  37. package/roborockLib/lib/message.js +223 -0
  38. package/roborockLib/lib/messageQueueHandler.js +87 -0
  39. package/roborockLib/lib/roborockPackageHelper.js +116 -0
  40. package/roborockLib/lib/roborock_mqtt_connector.js +349 -0
  41. package/roborockLib/lib/sniffing/mitmproxy_roborock.py +300 -0
  42. package/roborockLib/lib/vacuum.js +636 -0
  43. package/roborockLib/roborockAPI.js +1365 -0
  44. package/roborockLib/test.js +31 -0
  45. package/roborockLib/userdata.json +24 -0
@@ -0,0 +1,692 @@
1
+ "use strict";
2
+
3
+ const { createCanvas } = require("canvas");
4
+ const { Image } = require("canvas");
5
+
6
+ // Farben ändern
7
+ const colors = {
8
+ floor: "#23465e",
9
+ obstacle: "#2b2e30",
10
+ path: "rgba(255,255,255,0.5)",
11
+ newmap: true,
12
+ };
13
+ const orgcolors = [
14
+ "#017E82",
15
+ "#BD7B00",
16
+ "#C05A41",
17
+ "#4579B5",
18
+ "#434242", // wall
19
+ "#dfdfdf", // desel floor
20
+ ];
21
+ const offset = 60;
22
+
23
+ const robot =
24
+ "";
25
+ const robot1 =
26
+ "";
27
+ const charger =
28
+ "";
29
+ const tank =
30
+ "";
31
+ const spaceship =
32
+ "";
33
+ const robot2 =
34
+ "";
35
+ const originalRobot =
36
+ "";
37
+ const go_to_pin =
38
+ "";
39
+
40
+ //const ctximg = canvasImg.getContext('2d');
41
+
42
+ const img = new Image(); // Create a new Image
43
+ img.src = originalRobot;
44
+
45
+ const img_charger = new Image();
46
+ img_charger.src = charger;
47
+
48
+ const obstacleTitles = {
49
+ 0: "Wire",
50
+ 1: "Pet waste",
51
+ 2: "Footwear",
52
+ 3: "Pedestal",
53
+ 4: "Pedestal",
54
+ 5: "Power strip",
55
+ 9: "Scale",
56
+ 10: "Fabric",
57
+ 18: "Dustpan",
58
+ 25: "Dustpan",
59
+ 26: "Bar",
60
+ 27: "Bar",
61
+ };
62
+
63
+ class MapCreator {
64
+ constructor(adapter) {
65
+ this.adapter = adapter;
66
+ this.scaleFactor = adapter.config.map_scale;
67
+ }
68
+
69
+ getX(dimensions, px) {
70
+ return (px * this.scaleFactor) % dimensions.width;
71
+ }
72
+ getY(dimensions, px) {
73
+ return dimensions.height - Math.floor(px / (dimensions.width / this.scaleFactor)) * this.scaleFactor - this.scaleFactor;
74
+ }
75
+
76
+ robotXtoPixelX(image, robotCoord) {
77
+ return (robotCoord - image.position.left) * this.scaleFactor - 2;
78
+ }
79
+ robotYtoPixelY(image, robotCoord) {
80
+ return (image.dimensions.height / this.scaleFactor + image.position.top - robotCoord) * this.scaleFactor - 2;
81
+ }
82
+
83
+ rotateCanvas(img, angleInDegrees) {
84
+ const canvasImg = createCanvas(img.width, img.height);
85
+ const ctx = canvasImg.getContext("2d");
86
+ const angleOffset = 90;
87
+ let angleInRadians = ((angleInDegrees - angleOffset) * Math.PI) / 180;
88
+ angleInRadians = ((angleInRadians + Math.PI) % (2 * Math.PI)) - Math.PI; // Normalize angle to -π to π
89
+
90
+ ctx.translate(img.width / 2, img.height / 2);
91
+ ctx.rotate(-angleInRadians);
92
+ ctx.translate(-img.width / 2, -img.height / 2);
93
+ ctx.drawImage(img, 0, 0);
94
+
95
+ return canvasImg;
96
+ }
97
+
98
+ drawLineBresenham(imageData, x1, y1, x2, y2) {
99
+ const pixels = imageData.data;
100
+
101
+ const dx = Math.abs(x2 - x1);
102
+ const dy = Math.abs(y2 - y1);
103
+ const sx = x1 < x2 ? 1 : -1;
104
+ const sy = y1 < y2 ? 1 : -1;
105
+ let err = dx - dy;
106
+
107
+ for(;;) {
108
+ // Setze Pixel im ImageData
109
+ if (x1 >= 0 && x1 < imageData.width && y1 >= 0 && y1 < imageData.height) { // handle out of bounds. lineto would already do this but we need to set pixels directly
110
+ const index = (x1 + y1 * imageData.width) * 4;
111
+ pixels[index] = 128; // r
112
+ pixels[index + 1] = 128; // g
113
+ pixels[index + 2] = 128; // b
114
+ pixels[index + 3] = 128; // a
115
+ }
116
+
117
+ if (x1 === x2 && y1 === y2) break;
118
+ const e2 = 2 * err;
119
+ if (e2 > -dy) {
120
+ err -= dy;
121
+ x1 += sx;
122
+ }
123
+ if (e2 < dx) {
124
+ err += dx;
125
+ y1 += sy;
126
+ }
127
+ }
128
+ }
129
+
130
+ areRoomsAdjacent(roomA, roomB) {
131
+ const horizontalOverlap = roomA.minX <= roomB.maxX && roomA.maxX >= roomB.minX;
132
+ const verticalOverlap = roomA.minY <= roomB.maxY && roomA.maxY >= roomB.minY;
133
+ return horizontalOverlap && verticalOverlap;
134
+ }
135
+
136
+ hexToRGBA(hex, alpha = 1) {
137
+ const r = parseInt(hex.slice(1, 3), 16);
138
+ const g = parseInt(hex.slice(3, 5), 16);
139
+ const b = parseInt(hex.slice(5, 7), 16);
140
+ return `rgba(${r}, ${g}, ${b}, ${alpha})`;
141
+ }
142
+
143
+ canvasMap(mapdata, duid, selectedMap, mappedRooms, options) {
144
+ if (options) {
145
+ colors.floor = options.FLOORCOLOR;
146
+ colors.obstacle = options.WALLCOLOR;
147
+ colors.path = options.PATHCOLOR;
148
+ colors.newmap = options && options.newmap ? options.newmap : true;
149
+ if (options.ROBOT === "robot") {
150
+ img.src = robot;
151
+ } else if (options.ROBOT === "robot1") {
152
+ img.src = robot1;
153
+ } else if (options.ROBOT === "tank") {
154
+ img.src = tank;
155
+ } else if (options.ROBOT === "spaceship") {
156
+ img.src = spaceship;
157
+ } else if (options.ROBOT === "robot2") {
158
+ img.src = robot2;
159
+ } else if (options.ROBOT === "originalRobot") {
160
+ img.src = originalRobot;
161
+ }
162
+ }
163
+ let maxtop = 0;
164
+ let maxleft = 0;
165
+ let minleft = 0;
166
+ let mintop = 0;
167
+
168
+ mapdata.IMAGE.dimensions.width = mapdata.IMAGE.dimensions.width * this.scaleFactor;
169
+ mapdata.IMAGE.dimensions.height = mapdata.IMAGE.dimensions.height * this.scaleFactor;
170
+
171
+ const canvas = createCanvas(mapdata.IMAGE.dimensions.width, mapdata.IMAGE.dimensions.height);
172
+ const ctx = canvas.getContext("2d");
173
+
174
+ if (mapdata.IMAGE.pixels.floor && mapdata.IMAGE.pixels.floor.length !== 0) {
175
+ if (typeof mapdata.IMAGE.pixels.floor[0] === "number") {
176
+ // init min
177
+ minleft = mapdata.IMAGE.pixels.floor[0] % mapdata.IMAGE.dimensions.width;
178
+ mintop = mapdata.IMAGE.dimensions.height - 1 - Math.floor(mapdata.IMAGE.pixels.floor[0] / mapdata.IMAGE.dimensions.width);
179
+
180
+ ["floor", "obstacle"].forEach((key) => {
181
+ ctx.beginPath();
182
+ mapdata.IMAGE.pixels[key].forEach((px) => {
183
+ if (key === "obstacle") {
184
+ ctx.fillStyle = colors.newmap ? orgcolors[4] : colors.obstacle;
185
+ } else {
186
+ ctx.fillStyle = colors.newmap ? orgcolors[5] : colors.floor;
187
+ }
188
+ //ctx.fillStyle = colors[key];
189
+ ctx.rect(this.getX(mapdata.IMAGE.dimensions, px), this.getY(mapdata.IMAGE.dimensions, px), this.scaleFactor, this.scaleFactor);
190
+
191
+ maxtop = Math.max(maxtop, this.getY(mapdata.IMAGE.dimensions, px));
192
+ maxleft = Math.max(maxleft, this.getX(mapdata.IMAGE.dimensions, px));
193
+ minleft = Math.min(minleft, this.getX(mapdata.IMAGE.dimensions, px));
194
+ mintop = Math.min(mintop, this.getY(mapdata.IMAGE.dimensions, px));
195
+ });
196
+ ctx.fill();
197
+ });
198
+ }
199
+ }
200
+
201
+ // Zeichne Alle Räume
202
+ const segmentsData = {};
203
+ const segmentsBounds = {};
204
+ const assignedColors = {};
205
+ const availableColors = [...orgcolors];
206
+
207
+ if (mapdata.IMAGE.pixels.segments && !mapdata.CURRENTLY_CLEANED_BLOCKS && colors.newmap) {
208
+ mapdata.IMAGE.pixels.segments.forEach((px) => {
209
+ const segnum = (px >> 21);
210
+ const x = this.getX(mapdata.IMAGE.dimensions, px & 0xfffff);
211
+ const y = this.getY(mapdata.IMAGE.dimensions, px & 0xfffff);
212
+
213
+ if (!segmentsData[segnum]) {
214
+ segmentsData[segnum] = { points: [], minX: x, maxX: x, minY: y, maxY: y };
215
+ } else {
216
+ const segment = segmentsData[segnum];
217
+ segment.points.push({ x, y });
218
+ segment.minX = Math.min(segment.minX, x);
219
+ segment.maxX = Math.max(segment.maxX, x);
220
+ segment.minY = Math.min(segment.minY, y);
221
+ segment.maxY = Math.max(segment.maxY, y);
222
+ }
223
+ });
224
+
225
+ Object.keys(segmentsData).forEach(segnum => {
226
+ const segment = segmentsData[segnum];
227
+ segmentsBounds[segnum] = {
228
+ minX: segment.minX,
229
+ maxX: segment.maxX,
230
+ minY: segment.minY,
231
+ maxY: segment.maxY
232
+ };
233
+ });
234
+
235
+ Object.keys(segmentsBounds).forEach(segnum => {
236
+ const currentBounds = segmentsBounds[segnum];
237
+ const adjacentSegments = Object.keys(segmentsBounds).filter(otherSegnum => {
238
+ const otherBounds = segmentsBounds[otherSegnum];
239
+ return segnum !== otherSegnum && this.areRoomsAdjacent(currentBounds, otherBounds);
240
+ });
241
+
242
+ const usedColors = adjacentSegments.map(adjSegnum => assignedColors[adjSegnum]);
243
+ const availableColor = availableColors.find(color => !usedColors.includes(color));
244
+
245
+ if (availableColor) {
246
+ assignedColors[segnum] = availableColor;
247
+ } else {
248
+ assignedColors[segnum] = availableColors[Math.floor(Math.random() * availableColors.length)];
249
+ }
250
+ });
251
+
252
+ Object.keys(segmentsData).forEach(segnum => {
253
+ const segment = segmentsData[segnum];
254
+ ctx.fillStyle = assignedColors[segnum] || availableColors[0];
255
+ ctx.beginPath();
256
+ segment.points.forEach(point => {
257
+ ctx.rect(point.x, point.y, this.scaleFactor, this.scaleFactor);
258
+ });
259
+ ctx.fill();
260
+ });
261
+
262
+ ctx.closePath();
263
+ }
264
+
265
+ if (mapdata.CURRENTLY_CLEANED_BLOCKS && colors.newmap) {
266
+ let segnum, lastcolor;
267
+ ctx.beginPath();
268
+ mapdata.IMAGE.pixels.segments.forEach((px) => {
269
+ segnum = px >> 21;
270
+ if (mapdata.CURRENTLY_CLEANED_BLOCKS.includes(segnum)) {
271
+ if (segnum !== lastcolor) {
272
+ ctx.fill();
273
+ ctx.beginPath();
274
+ ctx.fillStyle = orgcolors[segnum % 4];
275
+ lastcolor = segnum;
276
+ }
277
+ px = px & 0xfffff;
278
+ ctx.rect(this.getX(mapdata.IMAGE.dimensions, px), this.getY(mapdata.IMAGE.dimensions, px), this.scaleFactor, this.scaleFactor);
279
+ }
280
+ });
281
+ ctx.fill();
282
+ ctx.closePath();
283
+ }
284
+
285
+ // Zeichne Zonen active Zonen
286
+ if (mapdata.CURRENTLY_CLEANED_ZONES) {
287
+ if (mapdata.CURRENTLY_CLEANED_ZONES[0]) {
288
+ ctx.beginPath();
289
+ mapdata.CURRENTLY_CLEANED_ZONES.forEach((coord) => {
290
+ ctx.fillStyle = "rgba(46,139,87,0.1)";
291
+ ctx.fillRect(
292
+ this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50),
293
+ this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50),
294
+ this.robotXtoPixelX(mapdata.IMAGE, coord[2] / 50) - this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50),
295
+ this.robotYtoPixelY(mapdata.IMAGE, coord[3] / 50) - this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50)
296
+ );
297
+ ctx.strokeStyle = "#2e8b57";
298
+ ctx.lineWidth = 4;
299
+ ctx.strokeRect(
300
+ this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50),
301
+ this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50),
302
+ this.robotXtoPixelX(mapdata.IMAGE, coord[2] / 50) - this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50),
303
+ this.robotYtoPixelY(mapdata.IMAGE, coord[3] / 50) - this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50)
304
+ );
305
+ });
306
+ }
307
+ }
308
+
309
+ // Zeichne Teppich
310
+ if (mapdata.CARPET_MAP) {
311
+ const offset = 8 * this.scaleFactor;
312
+ ctx.fillStyle = "rgba(0,0,0,0.5)";
313
+
314
+ const imageData = ctx.getImageData(0, 0, mapdata.IMAGE.dimensions.width, mapdata.IMAGE.dimensions.height);
315
+ mapdata.CARPET_MAP.forEach((px) => {
316
+ const x2 = this.getX(mapdata.IMAGE.dimensions, px) - offset;
317
+ const y1 = this.getY(mapdata.IMAGE.dimensions, px);
318
+ const x1 = x2 + this.scaleFactor - 1;
319
+ const y2 = y1 + this.scaleFactor - 1;
320
+
321
+ this.drawLineBresenham(imageData, x1, y1, x2, y2);
322
+ });
323
+
324
+ // Zeichne das ganze ImageData auf einmal
325
+ ctx.putImageData(imageData, 0, 0);
326
+ }
327
+
328
+ // Male den Wischpfad
329
+ if (mapdata.PATH && mapdata.MOP_PATH) {
330
+ const mopOffset = -12; // i dont know why this offset?? Maybe the value from the end
331
+ if (mapdata.PATH.points && mapdata.PATH.points.length !== 0) {
332
+ let startX, startY; // this is needed to avoid weird spikes in sharp corners! don't remove this ever!
333
+
334
+ ctx.beginPath();
335
+ ctx.lineWidth = 7 * this.scaleFactor; // 7 makes the mop path look the same as on the Roborock app
336
+ ctx.lineCap = "round";
337
+ ctx.strokeStyle = "rgba(255,255,255,0.2)";
338
+
339
+ mapdata.PATH.points.forEach((coord, index) => {
340
+ if (mapdata.MOP_PATH && mapdata.MOP_PATH[index + mopOffset] !== 0) {
341
+ if (mapdata.MOP_PATH[index - 1 + mopOffset] === 0) {
342
+ startX = this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50);
343
+ startY = this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50);
344
+ ctx.moveTo(startX, startY);
345
+ if (mapdata.MOP_PATH[index + mopOffset] !== 1) {
346
+ // see value 9 and 12 in mop_path both in front of charger
347
+ }
348
+ } else if (mapdata.MOP_PATH[index - 1 + mopOffset] === 1) {
349
+ ctx.moveTo(startX, startY);
350
+ ctx.lineTo(this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50), this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50));
351
+ startX = this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50);
352
+ startY = this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50);
353
+ }
354
+ } else if (mapdata.MOP_PATH && mapdata.MOP_PATH[index + mopOffset] === 0) {
355
+ if (mapdata.MOP_PATH[index - 1 + mopOffset] !== 0) {
356
+ // do nothing ??
357
+ }
358
+ }
359
+ });
360
+ ctx.stroke();
361
+ ctx.closePath();
362
+ }
363
+ }
364
+
365
+ // Male den Pfad
366
+ if (mapdata.PATH) {
367
+ if (mapdata.PATH.points && mapdata.PATH.points.length !== 0) {
368
+ ctx.fillStyle = colors.path;
369
+ let first = true;
370
+ let cold1, cold2;
371
+
372
+ ctx.beginPath();
373
+ mapdata.PATH.points.forEach((coord) => {
374
+ if (first) {
375
+ (cold1 = this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50)),
376
+ (cold2 = this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50)),
377
+ ctx.fillRect(cold1, cold2, (1 * this.scaleFactor) / 2, (1 * this.scaleFactor) / 2);
378
+ first = false;
379
+ } else {
380
+ ctx.lineWidth = this.scaleFactor / 2;
381
+ ctx.strokeStyle = colors.path;
382
+
383
+ ctx.moveTo(cold1, cold2);
384
+ (cold1 = this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50)), (cold2 = this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50)), ctx.lineTo(cold1, cold2);
385
+ // ctx.stroke();
386
+ }
387
+ });
388
+ ctx.stroke();
389
+ ctx.closePath();
390
+ }
391
+ }
392
+
393
+ // Male geplanten Pfad
394
+ if (mapdata.GOTO_PREDICTED_PATH) {
395
+ if (mapdata.GOTO_PREDICTED_PATH.points && mapdata.GOTO_PREDICTED_PATH.points.length !== 0) {
396
+ let cold1, cold2;
397
+ ctx.lineWidth = (3 * this.scaleFactor) / 2;
398
+ ctx.strokeStyle = "rgba(255, 255, 255, 1)";
399
+ ctx.setLineDash([3 * this.scaleFactor, 3 * this.scaleFactor]);
400
+ ctx.lineCap = "round";
401
+ ctx.beginPath();
402
+ mapdata.GOTO_PREDICTED_PATH.points.forEach((coord, index) => {
403
+ if (index === 0) {
404
+ cold1 = this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50);
405
+ cold2 = this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50);
406
+ ctx.fillStyle = "rgba(255, 255, 255, 1)";
407
+ ctx.fillRect(cold1, cold2, (1 * this.scaleFactor) / 2, (1 * this.scaleFactor) / 2);
408
+ ctx.moveTo(cold1, cold2);
409
+ } else {
410
+ const newCold1 = this.robotXtoPixelX(mapdata.IMAGE, coord[0] / 50);
411
+ const newCold2 = this.robotYtoPixelY(mapdata.IMAGE, coord[1] / 50);
412
+ if (cold1 !== newCold1 || cold2 !== newCold2) {
413
+ ctx.lineTo(newCold1, newCold2);
414
+ cold1 = newCold1;
415
+ cold2 = newCold2;
416
+ }
417
+ }
418
+ });
419
+ ctx.stroke();
420
+ ctx.setLineDash([]);
421
+ ctx.lineCap = "butt";
422
+ }
423
+ }
424
+
425
+ if (mapdata.FORBIDDEN_ZONES) {
426
+ mapdata.FORBIDDEN_ZONES.forEach((zone) => {
427
+ const forbiddenMinX = Math.min(zone[0], zone[2], zone[4], zone[6]);
428
+ const forbiddenMinY = Math.min(zone[1], zone[3], zone[5], zone[7]);
429
+ const forbiddenMaxX = Math.max(zone[0], zone[2], zone[4], zone[6]);
430
+ const forbiddenMaxY = Math.max(zone[1], zone[3], zone[5], zone[7]);
431
+
432
+ const forbiddenSizeX = forbiddenMaxX - forbiddenMinX;
433
+ const forbiddenSizeY = forbiddenMaxY - forbiddenMinY;
434
+
435
+ ctx.fillStyle = "rgba(255, 0, 0, 0.5)";
436
+ ctx.fillRect(
437
+ this.robotXtoPixelX(mapdata.IMAGE, forbiddenMinX / 50),
438
+ this.robotYtoPixelY(mapdata.IMAGE, forbiddenMaxY / 50),
439
+ (forbiddenSizeX / 50) * this.scaleFactor,
440
+ (forbiddenSizeY / 50) * this.scaleFactor
441
+ );
442
+ ctx.lineWidth = (1 * this.scaleFactor) / 2;
443
+ ctx.strokeStyle = "rgba(255, 0, 0, 1)";
444
+ ctx.strokeRect(
445
+ this.robotXtoPixelX(mapdata.IMAGE, forbiddenMinX / 50),
446
+ this.robotYtoPixelY(mapdata.IMAGE, forbiddenMaxY / 50),
447
+ (forbiddenSizeX / 50) * this.scaleFactor,
448
+ (forbiddenSizeY / 50) * this.scaleFactor
449
+ );
450
+ });
451
+ }
452
+
453
+ if (mapdata.VIRTUAL_WALLS) {
454
+ mapdata.VIRTUAL_WALLS.forEach((wall) => {
455
+ const startX = this.robotXtoPixelX(mapdata.IMAGE, wall[0] / 50) + this.scaleFactor;
456
+ const startY = this.robotYtoPixelY(mapdata.IMAGE, wall[1] / 50) + this.scaleFactor;
457
+ const endX = this.robotXtoPixelX(mapdata.IMAGE, wall[2] / 50) + this.scaleFactor;
458
+ const endY = this.robotYtoPixelY(mapdata.IMAGE, wall[3] / 50) + this.scaleFactor;
459
+
460
+ // Calculate start end end of vector
461
+ let vecX = endX - startX;
462
+ let vecY = endY - startY;
463
+
464
+ // Normalize vector
465
+ const len = Math.sqrt(vecX * vecX + vecY * vecY);
466
+ vecX /= len;
467
+ vecY /= len;
468
+
469
+ // Line width
470
+ const lineWidth = 1 * this.scaleFactor;
471
+
472
+ // Adjust start and end of the line
473
+ const adjustedStartX = startX + vecX * (lineWidth / 2);
474
+ const adjustedStartY = startY + vecY * (lineWidth / 2);
475
+ const adjustedEndX = endX - vecX * (lineWidth / 2);
476
+ const adjustedEndY = endY - vecY * (lineWidth / 2);
477
+
478
+ ctx.lineWidth = lineWidth;
479
+ ctx.strokeStyle = "rgba(255, 0, 0, 1)";
480
+
481
+ ctx.beginPath();
482
+ ctx.moveTo(adjustedStartX, adjustedStartY);
483
+ ctx.lineTo(adjustedEndX, adjustedEndY);
484
+ ctx.stroke();
485
+ });
486
+ }
487
+
488
+ if (mapdata.NO_MOP_ZONE) {
489
+ mapdata.NO_MOP_ZONE.forEach((zone) => {
490
+ const noMopMinX = Math.min(zone[0], zone[2], zone[4], zone[6]);
491
+ const noMopMinY = Math.min(zone[1], zone[3], zone[5], zone[7]);
492
+ const noMopMaxX = Math.max(zone[0], zone[2], zone[4], zone[6]);
493
+ const noMopMaxY = Math.max(zone[1], zone[3], zone[5], zone[7]);
494
+
495
+ const noMopSizeX = noMopMaxX - noMopMinX;
496
+ const noMopSizeY = noMopMaxY - noMopMinY;
497
+
498
+ ctx.fillStyle = "rgba(0, 0, 255, 0.5)";
499
+ ctx.fillRect(
500
+ this.robotXtoPixelX(mapdata.IMAGE, noMopMinX / 50),
501
+ this.robotYtoPixelY(mapdata.IMAGE, noMopMaxY / 50),
502
+ (noMopSizeX / 50) * this.scaleFactor,
503
+ (noMopSizeY / 50) * this.scaleFactor
504
+ );
505
+ ctx.lineWidth = (1 * this.scaleFactor) / 2;
506
+ ctx.strokeStyle = "rgba(0, 0, 255, 1)";
507
+ ctx.strokeRect(
508
+ this.robotXtoPixelX(mapdata.IMAGE, noMopMinX / 50),
509
+ this.robotYtoPixelY(mapdata.IMAGE, noMopMaxY / 50),
510
+ (noMopSizeX / 50) * this.scaleFactor,
511
+ (noMopSizeY / 50) * this.scaleFactor
512
+ );
513
+ });
514
+ }
515
+
516
+ if (mapdata.OBSTACLES2) {
517
+ mapdata.OBSTACLES2.forEach((obstacle) => {
518
+ const obstacleType = obstacle[2];
519
+ const obstacleTitle = obstacleTitles[obstacleType] || "Unknown";
520
+ const obstacleConfidence = Math.round(obstacle[3] / 100);
521
+ const obstacleText = obstacleTitle[obstacleType] ? obstacleTitle + "(" + obstacleConfidence + "%)" : obstacleTitle;
522
+
523
+ const x = this.robotXtoPixelX(mapdata.IMAGE, obstacle[0] / 50);
524
+ const y = this.robotYtoPixelY(mapdata.IMAGE, obstacle[1] / 50);
525
+
526
+ ctx.fillStyle = "red";
527
+ ctx.beginPath();
528
+ ctx.arc(x, y, 5, 0, 2 * Math.PI);
529
+ ctx.fill();
530
+
531
+ // Set the text properties
532
+ ctx.font = "14px sans-serif";
533
+ ctx.fillStyle = "white";
534
+ ctx.textBaseline = "middle";
535
+ ctx.textAlign = "center";
536
+
537
+ // Calculate the text width and height
538
+ const textWidth = ctx.measureText(obstacleText).width;
539
+ const textHeight = parseInt(ctx.font, 10);
540
+
541
+ // Calculate the position and dimensions of the background rectangle
542
+ const padding = 5;
543
+ const borderRadius = 5;
544
+ const rectX = x - textWidth / 2 - padding;
545
+ const rectY = y + 5 + padding / 2;
546
+ const rectWidth = textWidth + 2 * padding;
547
+ const rectHeight = textHeight + padding;
548
+
549
+ // Draw the background rectangle with rounded corners
550
+ ctx.fillStyle = "red";
551
+ ctx.beginPath();
552
+ ctx.moveTo(rectX + borderRadius, rectY);
553
+ ctx.lineTo(rectX + rectWidth - borderRadius, rectY);
554
+ ctx.quadraticCurveTo(rectX + rectWidth, rectY, rectX + rectWidth, rectY + borderRadius);
555
+ ctx.lineTo(rectX + rectWidth, rectY + rectHeight - borderRadius);
556
+ ctx.quadraticCurveTo(rectX + rectWidth, rectY + rectHeight, rectX + rectWidth - borderRadius, rectY + rectHeight);
557
+ ctx.lineTo(rectX + borderRadius, rectY + rectHeight);
558
+ ctx.quadraticCurveTo(rectX, rectY + rectHeight, rectX, rectY + rectHeight - borderRadius);
559
+ ctx.lineTo(rectX, rectY + borderRadius);
560
+ ctx.quadraticCurveTo(rectX, rectY, rectX + borderRadius, rectY);
561
+ ctx.closePath();
562
+ ctx.fill();
563
+
564
+ // Draw the white text centered within the background rectangle
565
+ ctx.fillStyle = "white";
566
+ ctx.fillText(obstacleText, x, y + 5 + padding + textHeight / 2);
567
+ });
568
+ }
569
+
570
+ // Zeichne Ladestation wenn vorhanden
571
+ if (mapdata.CHARGER_LOCATION) {
572
+ if (mapdata.CHARGER_LOCATION.position[0] && mapdata.CHARGER_LOCATION.position[1]) {
573
+ ctx.beginPath();
574
+ const img_charger_rotated = this.rotateCanvas(img_charger, mapdata.CHARGER_LOCATION.angle);
575
+ ctx.drawImage(
576
+ img_charger_rotated,
577
+ this.robotXtoPixelX(mapdata.IMAGE, mapdata.CHARGER_LOCATION.position[0] / 50) - img_charger_rotated.width / 2,
578
+ this.robotYtoPixelY(mapdata.IMAGE, mapdata.CHARGER_LOCATION.position[1] / 50) - img_charger_rotated.height / 2,
579
+ img_charger_rotated.width,
580
+ img_charger_rotated.height
581
+ );
582
+ }
583
+ }
584
+
585
+ // Zeichne Roboter
586
+ if (mapdata.ROBOT_POSITION) {
587
+ if (mapdata.PATH.current_angle && mapdata.ROBOT_POSITION[0] && mapdata.ROBOT_POSITION[1]) {
588
+ ctx.beginPath();
589
+ const canvasImg = this.rotateCanvas(img, mapdata.PATH.current_angle);
590
+ ctx.drawImage(
591
+ canvasImg,
592
+ this.robotXtoPixelX(mapdata.IMAGE, mapdata.ROBOT_POSITION.position[0] / 50) - img.width / 4,
593
+ this.robotYtoPixelY(mapdata.IMAGE, mapdata.ROBOT_POSITION.position[1] / 50) - img.height / 4,
594
+ canvasImg.width / 2,
595
+ canvasImg.height / 2
596
+ );
597
+ } else {
598
+ const img_robot_rotated = this.rotateCanvas(img, mapdata.ROBOT_POSITION.angle);
599
+ ctx.drawImage(
600
+ img_robot_rotated,
601
+ this.robotXtoPixelX(mapdata.IMAGE, mapdata.ROBOT_POSITION.position[0] / 50) - img_robot_rotated.width / 4,
602
+ this.robotYtoPixelY(mapdata.IMAGE, mapdata.ROBOT_POSITION.position[1] / 50) - img_robot_rotated.height / 4,
603
+ img_robot_rotated.width / 2,
604
+ img_robot_rotated.height / 2
605
+ );
606
+ }
607
+ }
608
+
609
+ // Zeichne Zielposition wenn vorhanden
610
+ if (mapdata.GOTO_TARGET) {
611
+ const go_to_pin_img = new Image();
612
+ go_to_pin_img.src = go_to_pin;
613
+
614
+ if (mapdata.GOTO_TARGET[0] && mapdata.GOTO_TARGET[1]) {
615
+ ctx.beginPath();
616
+ ctx.drawImage(
617
+ go_to_pin_img,
618
+ this.robotXtoPixelX(mapdata.IMAGE, mapdata.GOTO_TARGET[0] / 50) - go_to_pin_img.width / 2,
619
+ this.robotYtoPixelY(mapdata.IMAGE, mapdata.GOTO_TARGET[1] / 50) - (go_to_pin_img.height + 6),
620
+ go_to_pin_img.width,
621
+ go_to_pin_img.height
622
+ );
623
+ }
624
+ }
625
+
626
+ // Draw room names with background
627
+ ctx.beginPath();
628
+ Object.keys(segmentsData).forEach((segnum) => {
629
+ const segment = segmentsData[segnum];
630
+
631
+ let roomName = "";
632
+
633
+ if (typeof selectedMap != "undefined") {
634
+ // cannot get room name if map is from map history
635
+
636
+ for (const mappedRoom of mappedRooms) {
637
+ const segmentID = mappedRoom[0];
638
+ const roomID = mappedRoom[1];
639
+
640
+ if (segmentID == segnum) {
641
+ roomName = this.adapter.roomIDs[roomID];
642
+ break;
643
+ }
644
+ }
645
+
646
+ if (roomName != "") {
647
+ const centerX = segment.minX + (segment.maxX - segment.minX) / 2;
648
+ const centerY = segment.minY + (segment.maxY - segment.minY) / 2;
649
+
650
+ ctx.font = "bold 16px Arial";
651
+ const textWidth = ctx.measureText(roomName).width;
652
+ const textHeight = 16;
653
+ const padding = 10;
654
+ const backgroundWidth = textWidth + 2 * padding;
655
+ const backgroundHeight = textHeight + 2 * padding;
656
+
657
+ // fake square for shadow
658
+ const imgdata = ctx.getImageData(centerX - backgroundWidth / 2, centerY - backgroundHeight / 2, backgroundWidth, backgroundHeight);
659
+ ctx.shadowOffsetX = 4;
660
+ ctx.shadowOffsetY = 4;
661
+ ctx.shadowBlur = 5;
662
+ ctx.shadowColor = "rgba(0, 0, 0, 0.5)";
663
+ ctx.fillStyle = "rgba(0, 0, 0, 0)";
664
+ ctx.fillRect(centerX - backgroundWidth / 2, centerY - backgroundHeight / 2, backgroundWidth, backgroundHeight);
665
+ ctx.putImageData(imgdata, centerX - backgroundWidth / 2, centerY - backgroundHeight / 2);
666
+
667
+ // draw actual square over fake square to have a sharp shadow
668
+ ctx.shadowColor = "transparent";
669
+ ctx.fillStyle = this.hexToRGBA(assignedColors[segnum], 0.75);
670
+ ctx.fillRect(centerX - backgroundWidth / 2, centerY - backgroundHeight / 2, backgroundWidth, backgroundHeight);
671
+
672
+ ctx.fillStyle = "black";
673
+ ctx.textAlign = "center";
674
+ ctx.textBaseline = "middle";
675
+ ctx.fillText(roomName, centerX, centerY);
676
+ }
677
+ }
678
+ });
679
+ ctx.closePath();
680
+
681
+ // crop image
682
+ const canvas_trimmed = createCanvas(maxleft - minleft + 2 * offset, maxtop - mintop + 2 * offset);
683
+ const ctx_trimmed = canvas_trimmed.getContext("2d");
684
+ const trimmed = ctx.getImageData(minleft - offset, mintop - offset, maxleft - minleft + 2 * offset, maxtop - mintop + offset);
685
+
686
+ ctx_trimmed.putImageData(trimmed, 0, 0);
687
+
688
+ return [canvas.toDataURL(), canvas_trimmed.toDataURL()];
689
+ }
690
+ }
691
+
692
+ module.exports = MapCreator;