UNPKG

homebridge-roborock-vacuum-update

Version:

Comprehensive Homebridge plugin for Roborock vacuum cleaners with full HomeKit integration including mopping, dock features, and advanced controls.

714 lines (593 loc) 34.2 kB
var mapBase64, selectedElement; var left, topMap, mapMinX, mapMinY, mapSizeX, mapSizeY, goToTarget = false; var zoomLevel = 0.55; var scaleFactor = 2; var zoomStep = 0.1; var minZoom = 0.5; var maxZoom = 3; var wheelZoom = 1; var panOffsetX = 0; var panOffsetY = 0; var isPanning = false; var startX = 0; var startY = 0; var popupX, popupY; var go_to_pin_image = ""; var goToPin = new Image(); var canvasOffsetX, canvasOffsetY; var data; var map; var popupTimeout; var timeoutStart; var selectedObstacleID; window.onload = function () { var popup = document.getElementById("popup"); var popupImage = document.getElementById("popup-image"); var triangle = document.getElementById("triangle"); var largePhoto = document.getElementById("largePhoto"); var largePhotoImage = document.getElementById("largePhoto-image"); var socket = new WebSocket("ws://" + window.location.hostname + ":7906"); socket.onopen = () => { console.log("Connected to WebSocket server"); socket.send(JSON.stringify({ command: "getRobots" })); }; socket.onmessage = (event) => { // console.log(`Received message: ${event.data}`); data = JSON.parse(event.data); var command = data.command; // console.log(`Received message command: ${command}`); switch (command) { case "robotList": console.log("Robot IDs: " + data.parameters); selectedElement = document.getElementById("robotSelect"); if (selectedElement.childElementCount == 0) { data.parameters.forEach((robot) => { var option = document.createElement("option"); option.value = robot[0]; option.text = robot[1]; selectedElement.add(option); }); getMap(selectedElement.value); // get map once for the selected robot selectedElement.addEventListener("change", (event) => { getMap(event.target.value); }); } break; case "map": map = data.map; console.log(`map:`, map); if (selectedElement.value == data.duid) { mapBase64 = data.base64; scaleFactor = data.scale; left = map.IMAGE.position.left; topMap = map.IMAGE.position.top; drawBackgroundImage(); } break; case "get_status": if (selectedElement.value == data.duid) { if (data.parameters.isCleaning) { // console.log("Cleaning in progress for robot: " + data.duid); startButton.disabled = true; stop.disabled = false; } else { // console.log("Cleaning not in progress for robot: " + data.duid); startButton.disabled = false; stop.disabled = true; } } break; } }; socket.onclose = () => { console.log("Disconnected from WebSocket server"); setTimeout(() => { window.onload(); }, 10000); }; function getMap(duid) { const commandGetMap = {}; commandGetMap.command = "getMap"; commandGetMap.duid = duid; socket.send(JSON.stringify(commandGetMap)); } const canvas = document.getElementById("myCanvas"); const ctx = canvas.getContext("2d"); ctx.imageSmoothingEnabled = false; const maxPanX = canvas.width / 2; const maxPanY = canvas.height / 2; canvasOffsetX = canvas.getBoundingClientRect().left; canvasOffsetY = canvas.getBoundingClientRect().top; console.log("canvasOffsetX: " + canvasOffsetX); console.log("canvasOffsetY: " + canvasOffsetY); async function localCoordsToRobotCoords(imagePoint) { const image = new Image(); image.src = mapBase64; const point = {}; await new Promise((resolve) => { image.onload = function () { point.x = Math.round((imagePoint.x + left) * 50.0); point.y = Math.round((image.height / scaleFactor + topMap - imagePoint.y) * 50.0); resolve(point); }; }); return point; } function mouseXtoCanvasX(e) { return (e.pageX || e.touches[0].clientX) - canvasOffsetX - 2; } function mouseYtoCanvasY(e) { return (e.pageY || e.touches[0].clientY) - canvasOffsetY - 2; } function canvasXtoMapX(x) { return roundTwoDecimals(((x - panOffsetX) / wheelZoom / zoomLevel + mapMinX) / scaleFactor); } function canvasYtoMapY(y) { return roundTwoDecimals(((y - panOffsetY) / wheelZoom / zoomLevel + mapMinY) / scaleFactor); } // maybe I need the functions one day // function mapXtoCanvasX(x) { // return roundTwoDecimals((x * scaleFactor - mapMinX) * wheelZoom * zoomLevel); // } // function mapYtoCanvasY(y) { // return roundTwoDecimals((y * scaleFactor - mapMinY) * wheelZoom * zoomLevel); // } function getOriginalX(transformedX) { return Math.floor((transformedX - map.IMAGE.position.left) * scaleFactor - 2 - mapMinX); } function getOriginalY(transformedY) { return Math.floor((map.IMAGE.dimensions.height / scaleFactor + map.IMAGE.position.top - transformedY) * scaleFactor - 2 - mapMinY); } function rectXtoRobotX(x) { return Math.round(canvasXtoMapX((x + panOffsetX / wheelZoom) * wheelZoom)); } function rectYtoRobotY(y) { return Math.round(canvasYtoMapY((y + panOffsetY / wheelZoom) * wheelZoom)); } let rects = []; let zones = []; let isDragging = false; let isResizing = false; let selectedRect = null; let offsetRectX, offsetRectY; let redDotOffsetX, redDotOffsetY; async function downEvent(e) { const mouseX = mouseXtoCanvasX(e); const mouseY = mouseYtoCanvasY(e); console.log(`mouseX: ${mouseX} mouseY: ${mouseY}`); const mapX = canvasXtoMapX(mouseX); const mapY = canvasYtoMapY(mouseY); const point = await localCoordsToRobotCoords({ x: mapX, y: mapY }); // console.log("mousedown robot pos: " + JSON.stringify(point)); console.log("Robot coords: " + JSON.stringify([point.x, point.y])); if (goToTarget == true) { const data = {}; data.duid = selectedElement.value; data.command = "app_goto_target"; data.parameters = [point.x, point.y]; socket.send(JSON.stringify(data)); goToTarget = false; drawMap(true); } for (let i = 0; i < rects.length; i++) { const rect = rects[i]; const canvasMinRectX = Math.round((rect.x + panOffsetX / wheelZoom) * wheelZoom); const canvasMinRectY = Math.round((rect.y + panOffsetY / wheelZoom) * wheelZoom); const canvasMaxRectX = Math.round((rect.x + rect.width + panOffsetX / wheelZoom) * wheelZoom); const canvasMaxRectY = Math.round((rect.y + rect.height + panOffsetY / wheelZoom) * wheelZoom); offsetRectX = mouseX - canvasMinRectX; offsetRectY = mouseY - canvasMinRectY; redDotOffsetX = (mouseX - canvasMaxRectX) / wheelZoom; redDotOffsetY = (mouseY - canvasMaxRectY) / wheelZoom; const distanceRedDot = Math.sqrt(Math.pow(redDotOffsetX, 2) + Math.pow(redDotOffsetY, 2)); if (distanceRedDot < 10) { console.log("Resizing"); selectedRect = i; isResizing = true; e.preventDefault(); } else if (offsetRectX >= 0 && offsetRectY >= 0 && offsetRectX <= rect.width * wheelZoom + 1 && offsetRectY <= rect.height * wheelZoom + 1) { console.log("Square selected: " + i); selectedRect = i; isDragging = true; e.preventDefault(); break; } } if (!isResizing && !isDragging) { isPanning = true; startX = mouseX - panOffsetX; startY = mouseY - panOffsetY; console.log("startX: " + startX + " startY: " + startY); drawMap(true); } if (map?.OBSTACLES2) { map?.OBSTACLES2?.forEach((obstacle) => { const canvasX = getOriginalX(obstacle[0] / 50) * zoomLevel * wheelZoom + panOffsetX; const canvasY = getOriginalY(obstacle[1] / 50) * zoomLevel * wheelZoom + panOffsetY; const x = obstacle[0] / 50; const y = obstacle[1] / 50; const distance = Math.sqrt(Math.pow(mouseX - canvasX, 2) + Math.pow(mouseY - canvasY, 2)); console.log(`distance: ${distance}`); if (distance <= 5) { selectedObstacleID = obstacle[6]; if (popupTimeout) { clearTimeout(popupTimeout); popupTimeout = null; } console.log(`obstacle coords x: ${x}, y: ${y}`); popupX = x; popupY = y; const data = {}; data.duid = selectedElement.value; data.command = "get_photo"; data.attribute = { data_filter: { img_id: selectedObstacleID, type: 1, // 1 = small photo, 0 full photo }, }; if (popupImage.src) popupImage.src = ""; triangle.style.left = "45px"; triangle.style.top = "100px"; popup.style.display = "block"; socket.send(JSON.stringify(data)); socket.onmessage = function (event) { const serverData = JSON.parse(event.data); if (serverData.image) { popupImage.src = serverData.image; socket.onmessage = null; popupImage.addEventListener("click", function () { console.log(`Request to large photo started!`); const data = {}; data.duid = selectedElement.value; data.command = "get_photo"; data.attribute = { data_filter: { img_id: selectedObstacleID, type: 0, // 1 = small photo, 0 full photo }, }; socket.send(JSON.stringify(data)); socket.onmessage = function (event) { const serverData = JSON.parse(event.data); if (serverData.image) { popup.style.display = "none"; largePhoto.style.display = "block"; largePhotoImage.src = serverData.image; largePhotoImage.onclick = function () { largePhoto.style.display = "none"; }; socket.onmessage = null; } }; setTimeout(() => { largePhoto.style.display = "none"; }, 10000); }); } }; // Hide the popup after 10 seconds timeoutStart = Date.now(); popupTimeout = setTimeout(() => { popup.style.display = "none"; socket.onmessage = null; popupTimeout = null; selectedObstacleID = null; }, 10000); updatePopupPosition(); isPanning = false; } }); } drawMap(true); } canvas.addEventListener("mousedown", downEvent); canvas.addEventListener("touchstart", downEvent); function upEvent(e) { isDragging = false; isResizing = false; isPanning = false; if (rects.length > 0) { updateRobotZones(); } // This is needed to prevent the popup from hiding instantly after showing it const elapsed = Date.now() - timeoutStart; if (elapsed > 250) popup.style.display = "none"; e.preventDefault(false); } canvas.addEventListener("mouseup", upEvent); canvas.addEventListener("touchend", upEvent); canvas.addEventListener("mouseleave", () => { isPanning = false; }); let goToX = 0; let goToY = 0; async function moveEvent(e) { const mouseX = mouseXtoCanvasX(e); const mouseY = mouseYtoCanvasY(e); if (isDragging) { rects[selectedRect].x = roundTwoDecimals((mouseX - panOffsetX - offsetRectX) / wheelZoom); rects[selectedRect].y = roundTwoDecimals((mouseY - panOffsetY - offsetRectY) / wheelZoom); } else if (isResizing) { rects[selectedRect].width = roundTwoDecimals((mouseX - panOffsetX - rects[selectedRect].x - redDotOffsetX) / wheelZoom); rects[selectedRect].height = roundTwoDecimals((mouseY - panOffsetY - rects[selectedRect].y - redDotOffsetY) / wheelZoom); } else if (goToTarget) { goToX = (mouseX - panOffsetX) / wheelZoom; goToY = (mouseY - panOffsetY) / wheelZoom; } else if (isPanning) { const deltaX = roundTwoDecimals(mouseX - startX); const deltaY = roundTwoDecimals(mouseY - startY); // Calculate the limits based on the map dimensions and zoom level const minOffsetX = maxPanX - mapSizeX * zoomLevel * wheelZoom; const maxOffsetX = maxPanX; const minOffsetY = maxPanY - mapSizeY * zoomLevel * wheelZoom; const maxOffsetY = maxPanY; // Clamp the pan offsets within the defined boundaries panOffsetX = clamp(deltaX, minOffsetX, maxOffsetX); panOffsetY = clamp(deltaY, minOffsetY, maxOffsetY); updatePopupPosition(); } } canvas.addEventListener("mousemove", moveEvent); canvas.addEventListener("touchmove", moveEvent); canvas.addEventListener("wheel", (e) => { e.preventDefault(); const clientX = mouseXtoCanvasX(e); const clientY = mouseYtoCanvasY(e); const prevZoom = wheelZoom; if (e.deltaY < 0) { wheelZoom += zoomStep; } else { wheelZoom -= zoomStep; } wheelZoom = roundTwoDecimals(Math.min(Math.max(wheelZoom, minZoom), maxZoom) * 100) / 100; panOffsetX = roundTwoDecimals(clientX + panOffsetX - (clientX * wheelZoom) / prevZoom); panOffsetY = roundTwoDecimals(clientY + panOffsetY - (clientY * wheelZoom) / prevZoom); updatePopupPosition(); drawMap(true); if (rects.length > 0) { updateRobotZones(); } console.log(`wheelZoom: ${wheelZoom}`); }); const deleteButton = document.getElementById("deleteButton"); deleteButton.addEventListener("click", function () { if (rects[selectedRect]) { rects.splice(selectedRect, 1); console.log("rects.length: " + rects.length); if (rects.length < 5) { addButton.disabled = false; } if (rects.length < 1) { deleteButton.disabled = true; } selectedRect = rects.length - 1; drawMap(true); } }); const addButton = document.getElementById("addButton"); addButton.addEventListener("click", function () { goToTarget = false; const width = (25 * scaleFactor) / wheelZoom; const height = (25 * scaleFactor) / wheelZoom; const x = canvas.width / 2 - width / 2; const y = canvas.height / 2 - height / 2; rects.push({ x: x, y: y, width: width, height: height }); console.log("length: " + rects.length); if (rects.length > 0) { deleteButton.disabled = false; } if (rects.length > 4) { addButton.disabled = true; } console.log("Square spawned at: " + x + ":" + y); selectedRect = rects.length - 1; drawMap(true); updateRobotZones(); }); const startButton = document.getElementById("startButton"); startButton.addEventListener("click", function () { const data = {}; data.duid = selectedElement.value; if (zones.length > 0) { data.command = "app_zoned_clean"; data.parameters = zones; } else { data.command = "app_start"; } console.log("Zones to start with: " + JSON.stringify(zones)); socket.send(JSON.stringify(data)); rects = []; drawMap(true); startButton.style.display = "none"; pauseButton.style.display = "inline-block"; }); const pauseButton = document.getElementById("pauseButton"); pauseButton.addEventListener("click", function () { const data = {}; data.duid = selectedElement.value; data.command = "app_pause"; socket.send(JSON.stringify(data)); startButton.style.display = "inline-block"; pauseButton.style.display = "none"; }); const stop = document.getElementById("stopButton"); stop.addEventListener("click", function () { data.duid = selectedElement.value; console.log(map); data.command = "app_stop"; socket.send(JSON.stringify(data)); startButton.style.display = "inline-block"; pauseButton.style.display = "none"; }); const dock = document.getElementById("dockButton"); dock.addEventListener("click", function () { const data = {}; data.duid = selectedElement.value; data.command = "app_charge"; socket.send(JSON.stringify(data)); }); const goTo = document.getElementById("goToButton"); goTo.addEventListener("click", function () { goToTarget = true; goToPin.src = go_to_pin_image; drawMap(true); }); const resetZoom = document.getElementById("resetZoomButton"); resetZoom.addEventListener("click", function () { panOffsetX = 0; panOffsetY = 0; wheelZoom = 1; updatePopupPosition(); drawMap(true); }); const image = new Image(); const tempCanvas = document.createElement("canvas"); async function drawBackgroundImage() { // console.log("mapBase64:" + mapBase64); image.src = mapBase64; image.onload = await function () { const tempCtx = tempCanvas.getContext("2d", { willReadFrequently: true }); let mapMaxX = 0, mapMaxY = 0; mapMinX = image.width; mapMinY = image.height; tempCanvas.width = image.width; tempCanvas.height = image.height; tempCtx.drawImage(image, 0, 0); // Get the image data and calculate the actual dimensions of the image const imageData = tempCtx.getImageData(0, 0, image.width, image.height); const pixels = imageData.data; for (let i = 0; i < pixels.length; i += 4) { const alpha = pixels[i + 3]; if (alpha > 0) { // Check if the alpha value is non-zero const x = (i / 4) % image.width; const y = Math.floor(i / 4 / image.width); mapMinX = Math.min(mapMinX, x); mapMinY = Math.min(mapMinY, y); mapMaxX = Math.max(mapMaxX, x); mapMaxY = Math.max(mapMaxY, y); } } // Add some padding to the map mapMinX--; mapMinY--; mapMaxX++; mapMaxY++; mapSizeX = mapMaxX - mapMinX; mapSizeY = mapMaxY - mapMinY; const aspectRatio = canvas.width / canvas.height; const contentAspectRatio = mapSizeX / mapSizeY; if (contentAspectRatio > aspectRatio) { console.log("Aspect ratio is greater than canvas aspect ratio mapSizeX: " + mapSizeX + " mapSizeY: " + mapSizeY); zoomLevel = Math.round((canvas.width * 100) / mapSizeX) / 100; } else { console.log("Aspect ratio is less than canvas aspect ratio mapSizeX: " + mapSizeX + " mapSizeY: " + mapSizeY); zoomLevel = Math.round((canvas.height * 100) / mapSizeY) / 100; } console.log("zoomLevel: " + zoomLevel); drawMap(true); }; } function updateRobotZones() { // console.log("rect.x : " + rect.x + " rect.y: " + rect.y); // console.log("rect.x : " + rect.x + " rect.y: " + rect.y); // console.log("canvasOffsetX : " + canvasOffsetX + " canvasOffsetY: " + canvasOffsetY); // console.log("panOffsetX : " + panOffsetX + " panOffsetY: " + panOffsetY); // console.log("x : " + getX(rect.x) + " y: " + getY(rect.y)); // console.log("rect.x : " + rect.x + " rect.y: " + rect.y); zones = []; for (const rect in rects) { const zone = []; // const x = canvasXtoMapX(rects[rect].x); const x = rectXtoRobotX(rects[rect].x); const y = rectYtoRobotY(rects[rect].y); const xMax = rectXtoRobotX(rects[rect].x + rects[rect].width); const yMax = rectYtoRobotY(rects[rect].y + rects[rect].height); localCoordsToRobotCoords({ x: x, y: y }).then((coords1) => { console.log("coords1.x : " + coords1.x + " coords1.y: " + coords1.y); localCoordsToRobotCoords({ x: xMax, y: yMax }).then((coords2) => { console.log("coords2.x : " + coords2.x + " coords2.y: " + coords2.y); zone.push(Math.min(coords1.x, coords2.x)); zone.push(Math.min(coords1.y, coords2.y)); zone.push(Math.max(coords1.x, coords2.x)); zone.push(Math.max(coords1.y, coords2.y)); zone.push(parseInt(document.getElementById("cleanCount").value)); // clean count zones.push(zone); // console.log("zone: " + JSON.stringify(zone)); console.log("Zones length: " + zones.length); console.log("Zones to start with: " + JSON.stringify(zones)); }); }); } } function updatePopupPosition() { if (popupTimeout) { const x = getOriginalX(popupX) * zoomLevel * wheelZoom + panOffsetX; const y = getOriginalY(popupY) * zoomLevel * wheelZoom + panOffsetY; popup.style.left = `${x - parseInt(popupImage.style.width, 10) / 2 + 10}px`; popup.style.top = `${y - parseInt(popupImage.style.height, 10)}px`; } } // Create a temporary canvas to draw the truncated image const truncatedCanvas = document.createElement("canvas"); let rafId = null; function drawMap(manualDraw = false) { if (manualDraw || isDragging || isResizing || isPanning || goToTarget) { truncatedCanvas.width = mapSizeX; truncatedCanvas.height = mapSizeY; const truncatedCtx = truncatedCanvas.getContext("2d"); truncatedCtx.drawImage(image, mapMinX, mapMinY, mapSizeX, mapSizeY, 0, 0, mapSizeX, mapSizeY); ctx.clearRect(0, 0, canvas.width, canvas.height); ctx.save(); ctx.translate(panOffsetX, panOffsetY); ctx.scale(wheelZoom * zoomLevel, wheelZoom * zoomLevel); ctx.drawImage(truncatedCanvas, 0, 0, mapSizeX, mapSizeY); for (let i = 0; i < rects.length; i++) { const rect = rects[i]; ctx.fillStyle = "rgba(255, 255, 255, 0.5)"; ctx.fillRect(rect.x / zoomLevel, rect.y / zoomLevel, rect.width / zoomLevel, rect.height / zoomLevel); ctx.lineWidth = 5; ctx.strokeStyle = "rgba(255, 255, 255, 1)"; ctx.strokeRect(rect.x / zoomLevel, rect.y / zoomLevel, rect.width / zoomLevel, rect.height / zoomLevel); ctx.fillStyle = "red"; ctx.beginPath(); ctx.arc((rect.x + rect.width) / zoomLevel, (rect.y + rect.height) / zoomLevel, 10, 0, 2 * Math.PI); ctx.fill(); console.log("Square position: " + JSON.stringify(rect)); } // ctx.beginPath(); // ctx.arc(100, 100, 50, 0, 2 * Math.PI); // ctx.fillStyle = "red"; // ctx.fill(); // ctx.stroke(); const goToPosX = (goToX - 8) / zoomLevel; const goToPosY = (goToY - 12) / zoomLevel; const goToSizeX = goToPin.width / 2; const goToSizeY = goToPin.height / 2; ctx.drawImage(goToPin, goToPosX, goToPosY, goToSizeX, goToSizeY); ctx.restore(); // this is needed to update the canvas only when needed if (!rafId) { rafId = requestAnimationFrame(() => { rafId = null; drawMap(); }); } } } function roundTwoDecimals(number) { return Math.round(number * 100) / 100; } function clamp(value, min, max) { return Math.min(Math.max(value, min), max); } };