homebridge-roborock-vacuum 0.1.0 → 0.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "homebridge-roborock-vacuum",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.2",
|
|
4
4
|
"description": "Roborock Vacuum Cleaner - plugin for Homebridge.",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"keywords": [
|
|
@@ -35,7 +35,6 @@
|
|
|
35
35
|
"abstract-things": "0.9.0",
|
|
36
36
|
"axios": "^1.7.7",
|
|
37
37
|
"binary-parser": "^2.2.1",
|
|
38
|
-
"canvas": "^2.11.2",
|
|
39
38
|
"chalk": "4.1.2",
|
|
40
39
|
"crc-32": "^1.2.2",
|
|
41
40
|
"debug": "4.3.5",
|
|
@@ -1,4 +1,4 @@
|
|
|
1
1
|
{
|
|
2
|
-
"val": "{\"uid\":1466357,\"tokentype\":\"\",\"token\":\"d8fe0bd712334ee585bc2145140f07d1-1JZxT5AINkUhb5JDyL8S+w==\",\"rruid\":\"rr6238c899155830\",\"region\":\"us\",\"countrycode\":\"886\",\"country\":\"TW\",\"nickname\":\"tasict\",\"rriot\":{\"u\":\"1RQsQJ2o8bxMCo6F45pICu\",\"s\":\"
|
|
2
|
+
"val": "{\"uid\":1466357,\"tokentype\":\"\",\"token\":\"d8fe0bd712334ee585bc2145140f07d1-1JZxT5AINkUhb5JDyL8S+w==\",\"rruid\":\"rr6238c899155830\",\"region\":\"us\",\"countrycode\":\"886\",\"country\":\"TW\",\"nickname\":\"tasict\",\"rriot\":{\"u\":\"1RQsQJ2o8bxMCo6F45pICu\",\"s\":\"l5UDP7\",\"h\":\"7h2O8mNvkt\",\"k\":\"kxF1J95i\",\"r\":{\"r\":\"US\",\"a\":\"https://api-us.roborock.com\",\"m\":\"ssl://mqtt-us.roborock.com:8883\",\"l\":\"https://wood-us.roborock.com\"}},\"tuyaDeviceState\":0,\"avatarurl\":\"https://files.roborock.com/iottest/default_avatar.png\"}",
|
|
3
3
|
"ack": true
|
|
4
4
|
}
|
|
@@ -2,7 +2,6 @@
|
|
|
2
2
|
|
|
3
3
|
const rrMessage = require("./message").message;
|
|
4
4
|
const RRMapParser = require("./RRMapParser");
|
|
5
|
-
const MapCreator = require("./mapCreator");
|
|
6
5
|
const fs = require("fs");
|
|
7
6
|
const zlib = require("zlib");
|
|
8
7
|
|
|
@@ -36,7 +35,6 @@ class vacuum {
|
|
|
36
35
|
this.message = new rrMessage(this.adapter);
|
|
37
36
|
|
|
38
37
|
this.mapParser = new RRMapParser(this.adapter);
|
|
39
|
-
this.mapCreator = new MapCreator(this.adapter);
|
|
40
38
|
|
|
41
39
|
this.parameterFolders = {
|
|
42
40
|
get_mop_mode: "deviceStatus",
|
|
@@ -50,94 +48,10 @@ class vacuum {
|
|
|
50
48
|
};
|
|
51
49
|
}
|
|
52
50
|
|
|
53
|
-
async getMap(duid) {
|
|
54
|
-
if (this.adapter.config.enable_map_creation) {
|
|
55
|
-
this.adapter.log.debug(`Update map`);
|
|
56
|
-
|
|
57
|
-
try {
|
|
58
|
-
// const map = await connector.sendRequest(duid, "get_map_v1", [], true);
|
|
59
|
-
// const map = await this.adapter.rr_mqtt_connector.sendRequest(duid, "get_map_v1", [], true);
|
|
60
|
-
const map = await this.adapter.messageQueueHandler.sendRequest(duid, "get_map_v1", [], true);
|
|
61
|
-
// this.adapter.log.debug(`Map received: ${map}`);
|
|
62
|
-
if (map != "retry") {
|
|
63
|
-
const mappedRooms = await this.adapter.messageQueueHandler.sendRequest(duid, "get_room_mapping", []);
|
|
64
|
-
|
|
65
|
-
// const deviceStatus = await this.adapter.messageQueueHandler.sendRequest(duid, "get_status", []);
|
|
66
|
-
const deviceStatus = await this.adapter.messageQueueHandler.sendRequest(duid, "get_prop", ["get_status"]);
|
|
67
|
-
const selectedMap = deviceStatus[0].map_status >> 2 ?? -1; // to get the currently selected map perform bitwise right shift
|
|
68
|
-
|
|
69
|
-
// This is for testing and debugging maps. This can't be stored in a state.
|
|
70
|
-
zlib.gzip(map, (error, buffer) => {
|
|
71
|
-
if (error) {
|
|
72
|
-
this.adapter.log.error(`Error compressing map to gz ${error}`);
|
|
73
|
-
} else {
|
|
74
|
-
fs.writeFile("./test.rrmap.gz", buffer, (error) => {
|
|
75
|
-
if (error) {
|
|
76
|
-
this.adapter.log.error(`Error writing map file ${error}`);
|
|
77
|
-
}
|
|
78
|
-
});
|
|
79
|
-
}
|
|
80
|
-
});
|
|
81
|
-
|
|
82
|
-
const parsedData = await this.mapParser.parsedata(map);
|
|
83
|
-
|
|
84
|
-
const [mapBase64, mapBase64Truncated] = this.mapCreator.canvasMap(parsedData, duid, selectedMap, mappedRooms);
|
|
85
|
-
|
|
86
|
-
await this.adapter.setStateAsync(`Devices.${duid}.map.mapData`, { val: JSON.stringify(parsedData), ack: true });
|
|
87
|
-
await this.adapter.setStateAsync(`Devices.${duid}.map.mapBase64`, { val: mapBase64, ack: true });
|
|
88
|
-
await this.adapter.setStateAsync(`Devices.${duid}.map.mapBase64Truncated`, { val: mapBase64Truncated, ack: true });
|
|
89
|
-
|
|
90
|
-
// Send current map with Scale factor
|
|
91
|
-
const mapToSend = {
|
|
92
|
-
duid: duid,
|
|
93
|
-
command: "map",
|
|
94
|
-
base64: mapBase64,
|
|
95
|
-
map: parsedData,
|
|
96
|
-
scale: this.adapter.config.map_scale,
|
|
97
|
-
};
|
|
98
|
-
|
|
99
|
-
if (this.adapter.socket != null) {
|
|
100
|
-
this.adapter.socket.send(JSON.stringify(mapToSend));
|
|
101
|
-
}
|
|
102
|
-
}
|
|
103
|
-
} catch (error) {
|
|
104
|
-
this.adapter.catchError(error, "get_map_v1", duid), this.robotModel;
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
async getCleaningRecordMap(duid, startTime) {
|
|
110
|
-
try {
|
|
111
|
-
const cleaningRecordMap = await this.adapter.messageQueueHandler.sendRequest(duid, "get_clean_record_map", { start_time: startTime }, true);
|
|
112
|
-
const parsedData = await this.mapParser.parsedata(cleaningRecordMap);
|
|
113
|
-
const [mapBase64, mapBase64Truncated] = this.mapCreator.canvasMap(parsedData, duid);
|
|
114
|
-
|
|
115
|
-
return {
|
|
116
|
-
mapBase64: mapBase64,
|
|
117
|
-
mapBase64Truncated: mapBase64Truncated,
|
|
118
|
-
mapData: JSON.stringify(parsedData),
|
|
119
|
-
};
|
|
120
|
-
} catch (error) {
|
|
121
|
-
this.adapter.catchError(error, "get_clean_record_map", duid, this.robotModel);
|
|
122
|
-
|
|
123
|
-
return null;
|
|
124
|
-
}
|
|
125
|
-
}
|
|
126
51
|
|
|
127
52
|
async command(duid, parameter, value) {
|
|
128
53
|
try {
|
|
129
54
|
switch (parameter) {
|
|
130
|
-
case "load_multi_map": {
|
|
131
|
-
const result = await this.adapter.messageQueueHandler.sendRequest(duid, "load_multi_map", value);
|
|
132
|
-
|
|
133
|
-
if (result[0] == "ok") {
|
|
134
|
-
await this.getMap(duid).then(async () => {
|
|
135
|
-
await this.getParameter(duid, "get_room_mapping");
|
|
136
|
-
});
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
break;
|
|
140
|
-
}
|
|
141
55
|
case "app_segment_clean": {
|
|
142
56
|
this.adapter.log.debug("Start room cleaning");
|
|
143
57
|
|
|
@@ -304,7 +218,6 @@ class vacuum {
|
|
|
304
218
|
|
|
305
219
|
if (mapFromCommand && mapFromCommand.val != currentMap) {
|
|
306
220
|
await this.adapter.setStateAsync(`Devices.${duid}.commands.load_multi_map`, currentMap, true);
|
|
307
|
-
await this.adapter.vacuums[duid].getMap(duid);
|
|
308
221
|
}
|
|
309
222
|
}
|
|
310
223
|
}
|
|
@@ -549,13 +462,6 @@ class vacuum {
|
|
|
549
462
|
});
|
|
550
463
|
}
|
|
551
464
|
|
|
552
|
-
if (this.adapter.config.enable_map_creation == true) {
|
|
553
|
-
const mapArray = await this.getCleaningRecordMap(duid, cleaningAttributes[cleaningAttribute][cleaningRecord]);
|
|
554
|
-
for (const mapType in mapArray) {
|
|
555
|
-
const val = mapArray[mapType];
|
|
556
|
-
this.adapter.setStateAsync(`Devices.${duid}.cleaningInfo.records.${cleaningRecord}.map.${mapType}`, { val: val, ack: true });
|
|
557
|
-
}
|
|
558
|
-
}
|
|
559
465
|
}
|
|
560
466
|
|
|
561
467
|
const objectString = `Devices.${duid}.cleaningInfo.JSON`;
|
|
@@ -1,692 +0,0 @@
|
|
|
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;
|