Navigation

    Logo
    • Register
    • Login
    • Search
    • Recent
    • Tags
    • Unread
    • Categories
    • Unreplied
    • Popular
    • GitHub
    • Docu
    • Hilfe
    1. Home
    2. Deutsch
    3. Skripten / Logik
    4. JavaScript
    5. [gelöst] Mi Robot Map per Telegram

    NEWS

    • Neuer Blog: Fotos und Eindrücke aus Solingen

    • ioBroker@Smart Living Forum Solingen, 14.06. - Agenda added

    • ioBroker goes Matter ... Matter Adapter in Stable

    SOLVED [gelöst] Mi Robot Map per Telegram

    This topic has been deleted. Only users with topic management privileges can see it.
    • AxelF1977
      AxelF1977 last edited by AxelF1977

      Hallo zusammen,

      nachdem der Sauger erfolgreich gerootet wurde, die Karte in VIS läuft, der Status per Telegram übertragen wird, hätte ich jetzt auch die karte per Telegram übermittelt.

      Hat das schonmal geklapt? Hat das schon jemand umgesetzt?

      Danke

      1 Reply Last reply Reply Quote 0
      • Meistertr
        Meistertr Developer last edited by

        hiermmit gehts 🙂

        createState('vis.RockroboMap', '');
        
        var fs = require("fs");
        
        const {
            createCanvas,
            Canvas
        } = require('canvas')
        const {
            Image
        } = require('canvas')
        const request = require('request');
        const XMLHttpRequest = require("xmlhttprequest").XMLHttpRequest;
        
        //________________________________________________________________________________________________________________________________________________
        // Farben änder
        const COLOR_FLOOR = "#23465e";
        const COLOR_WALLS = "#2b2e30";
        const COLOR_PATH = "white";
        
        //________________________________________________________________________________________________________________________________________________
        
        
        
        //Robot Image
        const rocky = "";
        const charger = "";
        
        //last map
        let last_map;
        
        var canvasimg = new Canvas();
        var ctximg = canvasimg.getContext('2d');
        var res = {};
        
        var img = new Image(); // Create a new Image
        img.src = rocky
        
        const img_charger = new Image();
        img_charger.src = charger;
        
        const robotIp = "rockrobo"; // IP of the robot
        const robotState = "mihome-vacuum.0.info.state" // e.g : "mihome-vacuum.0.info.state"
        
        httpGetAsync("http://" + robotIp + "/api/map/latest", updateMapPage);
        
        
        // get actuel map data from Valetudo
        function httpGetAsync(theUrl, callback) {
            var xmlHttp = new XMLHttpRequest();
            var jdata = {};
            xmlHttp.onreadystatechange = function () {
                //console.log(JSON.stringify(xmlHttp));
                if (xmlHttp.readyState == 4 && xmlHttp.status == 200) {
                    try {
                        jdata = JSON.parse(xmlHttp.responseText)
                        //console.log(JSON.stringify(jdata));
                    } catch (err) {}
                    callback(jdata);
                }
        
            }
            xmlHttp.open("GET", theUrl, true); // true for asynchronous 
            xmlHttp.send(null);
        }
        
        function updateMapPage(res) {
            var canvas = createCanvas();
            var ctx = canvas.getContext('2d');
        
            let mapImageData;
            var map;
            canvas.height = 1024 * 4 //res.image.dimensions.height;
            canvas.width = 1024 * 4 //res.image.dimensions.width;
        
            // Male Boden
            if (res.image.pixels.floor && res.image.pixels.floor.length !== 0) {
                ctx.fillStyle = COLOR_FLOOR;
                res.image.pixels.floor.forEach(function (coord) {
                    ctx.fillRect(coord[0] * 4 + res.image.position.left * 4, coord[1] * 4 + res.image.position.top * 4, 4, 4);
        
                });
            }
            // Male Wände
            if (res.image.pixels.obstacle_strong && res.image.pixels.obstacle_strong.length !== 0) {
                ctx.fillStyle = COLOR_WALLS;
                res.image.pixels.obstacle_strong.forEach(function (coord) {
                    ctx.fillRect(coord[0] * 4 + res.image.position.left * 4, coord[1] * 4 + res.image.position.top * 4, 4, 4);
        
                });
            }
        
            // Male den Pfad
            if (res.path.points && res.path.points.length !== 0) {
                ctx.fillStyle = COLOR_PATH;
                let first = true;
                let cold1, cold2;
        
        
        
                res.path.points.forEach(function (coord) {
                    if (first) {
                        ctx.fillRect(coord[0] / 12.5, coord[1] / 50, 2, 2);
                        cold1 = coord[0] / 12.5;
                        cold2 = coord[1] / 12.5;
                    } else {
                        ctx.beginPath();
                        ctx.lineWidth = 1;
                        ctx.strokeStyle = "#FFFFFF";
                        ctx.moveTo(cold1, cold2);
                        ctx.lineTo(coord[0] / 12.5, coord[1] / 12.5);
                        ctx.stroke();
        
                        cold1 = coord[0] / 12.5
                        cold2 = coord[1] / 12.5
                    }
                    first = false
        
                });
            }
            // Zeichne Roboter
            ctx.beginPath();
            if (res.robot){
                if (res.path.current_angle && typeof res.robot[0] !== "undefined" && typeof res.robot[1] !== "undefined") {
                    canvasimg = rotateRobo(img, res.path.current_angle);
                    ctx.drawImage(canvasimg, res.robot[0] / 12.5 - 15, res.robot[1] / 12.5 - 15, img.width, img.height);
                } else {
                    ctx.drawImage(img, res.robot[0] / 12.5 - 15, res.robot[1] / 12.5 - 15, img.width, img.height);
                }
            }
            // Zeichne Ladestation wenn vorhanden
            if (res.charger){
                if (typeof res.charger[0] !== "undefined" && typeof res.charger[1] !== "undefined") {
                    ctx.beginPath();
                    ctx.drawImage(img_charger, res.charger[0] / 12.5 - 15, res.charger[1] / 12.5 - 15);
                }
            }
          
        
        
            // crop image
            let canvas_final = createCanvas();
            let ctx_final = canvas_final.getContext('2d');
            var trimmed = ctx.getImageData(res.image.position.left * 4, res.image.position.top * 4, res.image.dimensions.width * 4, res.image.dimensions.height * 4);
        
            canvas_final.height = res.image.dimensions.height * 4;
            canvas_final.width = res.image.dimensions.width * 4;
        
            ctx_final.putImageData(trimmed, 0, 0);
        
        
            map = canvas_final.toDataURL();
            last_map=  canvas_final;
            setState("javascript.0.vis.RockroboMap", '<img src="' + canvas_final.toDataURL() + '" /style="width: auto ;height: 100%;">');
            //log('<img src="' + canvas_final.toDataURL() + '" />');
        }
        function send(canvas){
        var buf = canvas.toBuffer();
        fs.writeFile("/opt/iobroker/vac_map.png", buf, (err) => {
          if (err) throw err;
          log('The file has been saved!');
        });
        setTimeout(function(){
           // sendTo('telegram.0', '/opt/iobroker/vac_map.png');
            sendTo('telegram.0', {text: '/opt/iobroker/vac_map.png', caption: 'Fertig mit saugen!'});
         }, 3000);
        
        }
        
        function rotateRobo(img, angle) {
            var canvasimg = createCanvas(img.width, img.height);
            var ctximg = canvasimg.getContext('2d');
            const offset = 90;
        
            ctximg.clearRect(0, 0, img.width, img.height);
            ctximg.translate(img.width / 2, img.width / 2);
            ctximg.rotate((angle + offset) * Math.PI / 180);
            ctximg.translate(-img.width / 2, -img.width / 2);
            ctximg.drawImage(img, 0, 0);
            return canvasimg;
        }
        schedule("*/2 * * * * *", function () {
            var robyState = getState(robotState).val;
        
            if (robyState === 5 || robyState === 11 || robyState === 17) httpGetAsync("http://" + robotIp + "/api/map/latest", updateMapPage);
        });
        
        subscribe({ id: 'mihome-vacuum.0.info.state'/*Vacuum state*/, change: "ne" }, function (obj) {
        
            if(obj.newState.val === 8 && obj.oldState.val !== 8){
            send(last_map);
            }
        });
        
        AxelF1977 1 Reply Last reply Reply Quote 0
        • Meistertr
          Meistertr Developer last edited by

          hiermmit gehts 🙂

          createState('vis.RockroboMap', '');
          
          var fs = require("fs");
          
          const {
              createCanvas,
              Canvas
          } = require('canvas')
          const {
              Image
          } = require('canvas')
          const request = require('request');
          const XMLHttpRequest = require("xmlhttprequest").XMLHttpRequest;
          
          //________________________________________________________________________________________________________________________________________________
          // Farben änder
          const COLOR_FLOOR = "#23465e";
          const COLOR_WALLS = "#2b2e30";
          const COLOR_PATH = "white";
          
          //________________________________________________________________________________________________________________________________________________
          
          
          
          //Robot Image
          const rocky = "";
          const charger = "";
          
          //last map
          let last_map;
          
          var canvasimg = new Canvas();
          var ctximg = canvasimg.getContext('2d');
          var res = {};
          
          var img = new Image(); // Create a new Image
          img.src = rocky
          
          const img_charger = new Image();
          img_charger.src = charger;
          
          const robotIp = "rockrobo"; // IP of the robot
          const robotState = "mihome-vacuum.0.info.state" // e.g : "mihome-vacuum.0.info.state"
          
          httpGetAsync("http://" + robotIp + "/api/map/latest", updateMapPage);
          
          
          // get actuel map data from Valetudo
          function httpGetAsync(theUrl, callback) {
              var xmlHttp = new XMLHttpRequest();
              var jdata = {};
              xmlHttp.onreadystatechange = function () {
                  //console.log(JSON.stringify(xmlHttp));
                  if (xmlHttp.readyState == 4 && xmlHttp.status == 200) {
                      try {
                          jdata = JSON.parse(xmlHttp.responseText)
                          //console.log(JSON.stringify(jdata));
                      } catch (err) {}
                      callback(jdata);
                  }
          
              }
              xmlHttp.open("GET", theUrl, true); // true for asynchronous 
              xmlHttp.send(null);
          }
          
          function updateMapPage(res) {
              var canvas = createCanvas();
              var ctx = canvas.getContext('2d');
          
              let mapImageData;
              var map;
              canvas.height = 1024 * 4 //res.image.dimensions.height;
              canvas.width = 1024 * 4 //res.image.dimensions.width;
          
              // Male Boden
              if (res.image.pixels.floor && res.image.pixels.floor.length !== 0) {
                  ctx.fillStyle = COLOR_FLOOR;
                  res.image.pixels.floor.forEach(function (coord) {
                      ctx.fillRect(coord[0] * 4 + res.image.position.left * 4, coord[1] * 4 + res.image.position.top * 4, 4, 4);
          
                  });
              }
              // Male Wände
              if (res.image.pixels.obstacle_strong && res.image.pixels.obstacle_strong.length !== 0) {
                  ctx.fillStyle = COLOR_WALLS;
                  res.image.pixels.obstacle_strong.forEach(function (coord) {
                      ctx.fillRect(coord[0] * 4 + res.image.position.left * 4, coord[1] * 4 + res.image.position.top * 4, 4, 4);
          
                  });
              }
          
              // Male den Pfad
              if (res.path.points && res.path.points.length !== 0) {
                  ctx.fillStyle = COLOR_PATH;
                  let first = true;
                  let cold1, cold2;
          
          
          
                  res.path.points.forEach(function (coord) {
                      if (first) {
                          ctx.fillRect(coord[0] / 12.5, coord[1] / 50, 2, 2);
                          cold1 = coord[0] / 12.5;
                          cold2 = coord[1] / 12.5;
                      } else {
                          ctx.beginPath();
                          ctx.lineWidth = 1;
                          ctx.strokeStyle = "#FFFFFF";
                          ctx.moveTo(cold1, cold2);
                          ctx.lineTo(coord[0] / 12.5, coord[1] / 12.5);
                          ctx.stroke();
          
                          cold1 = coord[0] / 12.5
                          cold2 = coord[1] / 12.5
                      }
                      first = false
          
                  });
              }
              // Zeichne Roboter
              ctx.beginPath();
              if (res.robot){
                  if (res.path.current_angle && typeof res.robot[0] !== "undefined" && typeof res.robot[1] !== "undefined") {
                      canvasimg = rotateRobo(img, res.path.current_angle);
                      ctx.drawImage(canvasimg, res.robot[0] / 12.5 - 15, res.robot[1] / 12.5 - 15, img.width, img.height);
                  } else {
                      ctx.drawImage(img, res.robot[0] / 12.5 - 15, res.robot[1] / 12.5 - 15, img.width, img.height);
                  }
              }
              // Zeichne Ladestation wenn vorhanden
              if (res.charger){
                  if (typeof res.charger[0] !== "undefined" && typeof res.charger[1] !== "undefined") {
                      ctx.beginPath();
                      ctx.drawImage(img_charger, res.charger[0] / 12.5 - 15, res.charger[1] / 12.5 - 15);
                  }
              }
            
          
          
              // crop image
              let canvas_final = createCanvas();
              let ctx_final = canvas_final.getContext('2d');
              var trimmed = ctx.getImageData(res.image.position.left * 4, res.image.position.top * 4, res.image.dimensions.width * 4, res.image.dimensions.height * 4);
          
              canvas_final.height = res.image.dimensions.height * 4;
              canvas_final.width = res.image.dimensions.width * 4;
          
              ctx_final.putImageData(trimmed, 0, 0);
          
          
              map = canvas_final.toDataURL();
              last_map=  canvas_final;
              setState("javascript.0.vis.RockroboMap", '<img src="' + canvas_final.toDataURL() + '" /style="width: auto ;height: 100%;">');
              //log('<img src="' + canvas_final.toDataURL() + '" />');
          }
          function send(canvas){
          var buf = canvas.toBuffer();
          fs.writeFile("/opt/iobroker/vac_map.png", buf, (err) => {
            if (err) throw err;
            log('The file has been saved!');
          });
          setTimeout(function(){
             // sendTo('telegram.0', '/opt/iobroker/vac_map.png');
              sendTo('telegram.0', {text: '/opt/iobroker/vac_map.png', caption: 'Fertig mit saugen!'});
           }, 3000);
          
          }
          
          function rotateRobo(img, angle) {
              var canvasimg = createCanvas(img.width, img.height);
              var ctximg = canvasimg.getContext('2d');
              const offset = 90;
          
              ctximg.clearRect(0, 0, img.width, img.height);
              ctximg.translate(img.width / 2, img.width / 2);
              ctximg.rotate((angle + offset) * Math.PI / 180);
              ctximg.translate(-img.width / 2, -img.width / 2);
              ctximg.drawImage(img, 0, 0);
              return canvasimg;
          }
          schedule("*/2 * * * * *", function () {
              var robyState = getState(robotState).val;
          
              if (robyState === 5 || robyState === 11 || robyState === 17) httpGetAsync("http://" + robotIp + "/api/map/latest", updateMapPage);
          });
          
          subscribe({ id: 'mihome-vacuum.0.info.state'/*Vacuum state*/, change: "ne" }, function (obj) {
          
              if(obj.newState.val === 8 && obj.oldState.val !== 8){
              send(last_map);
              }
          });
          
          AxelF1977 1 Reply Last reply Reply Quote 0
          • Meistertr
            Meistertr Developer last edited by

            Habs auch hochgeladen

            1 Reply Last reply Reply Quote 0
            • AxelF1977
              AxelF1977 @Meistertr last edited by

              @Meistertr sagte in Mi Robot Map per Telegram:

              hiermmit gehts 🙂

              createState('vis.RockroboMap', '');
              
              var fs = require("fs");
              
              const {
                  createCanvas,
                  Canvas
              } = require('canvas')
              const {
                  Image
              } = require('canvas')
              const request = require('request');
              const XMLHttpRequest = require("xmlhttprequest").XMLHttpRequest;
              
              //________________________________________________________________________________________________________________________________________________
              // Farben änder
              const COLOR_FLOOR = "#23465e";
              const COLOR_WALLS = "#2b2e30";
              const COLOR_PATH = "white";
              
              //________________________________________________________________________________________________________________________________________________
              
              
              
              //Robot Image
              const rocky = "";
              const charger = "";
              
              //last map
              let last_map;
              
              var canvasimg = new Canvas();
              var ctximg = canvasimg.getContext('2d');
              var res = {};
              
              var img = new Image(); // Create a new Image
              img.src = rocky
              
              const img_charger = new Image();
              img_charger.src = charger;
              
              const robotIp = "rockrobo"; // IP of the robot
              const robotState = "mihome-vacuum.0.info.state" // e.g : "mihome-vacuum.0.info.state"
              
              httpGetAsync("http://" + robotIp + "/api/map/latest", updateMapPage);
              
              
              // get actuel map data from Valetudo
              function httpGetAsync(theUrl, callback) {
                  var xmlHttp = new XMLHttpRequest();
                  var jdata = {};
                  xmlHttp.onreadystatechange = function () {
                      //console.log(JSON.stringify(xmlHttp));
                      if (xmlHttp.readyState == 4 && xmlHttp.status == 200) {
                          try {
                              jdata = JSON.parse(xmlHttp.responseText)
                              //console.log(JSON.stringify(jdata));
                          } catch (err) {}
                          callback(jdata);
                      }
              
                  }
                  xmlHttp.open("GET", theUrl, true); // true for asynchronous 
                  xmlHttp.send(null);
              }
              
              function updateMapPage(res) {
                  var canvas = createCanvas();
                  var ctx = canvas.getContext('2d');
              
                  let mapImageData;
                  var map;
                  canvas.height = 1024 * 4 //res.image.dimensions.height;
                  canvas.width = 1024 * 4 //res.image.dimensions.width;
              
                  // Male Boden
                  if (res.image.pixels.floor && res.image.pixels.floor.length !== 0) {
                      ctx.fillStyle = COLOR_FLOOR;
                      res.image.pixels.floor.forEach(function (coord) {
                          ctx.fillRect(coord[0] * 4 + res.image.position.left * 4, coord[1] * 4 + res.image.position.top * 4, 4, 4);
              
                      });
                  }
                  // Male Wände
                  if (res.image.pixels.obstacle_strong && res.image.pixels.obstacle_strong.length !== 0) {
                      ctx.fillStyle = COLOR_WALLS;
                      res.image.pixels.obstacle_strong.forEach(function (coord) {
                          ctx.fillRect(coord[0] * 4 + res.image.position.left * 4, coord[1] * 4 + res.image.position.top * 4, 4, 4);
              
                      });
                  }
              
                  // Male den Pfad
                  if (res.path.points && res.path.points.length !== 0) {
                      ctx.fillStyle = COLOR_PATH;
                      let first = true;
                      let cold1, cold2;
              
              
              
                      res.path.points.forEach(function (coord) {
                          if (first) {
                              ctx.fillRect(coord[0] / 12.5, coord[1] / 50, 2, 2);
                              cold1 = coord[0] / 12.5;
                              cold2 = coord[1] / 12.5;
                          } else {
                              ctx.beginPath();
                              ctx.lineWidth = 1;
                              ctx.strokeStyle = "#FFFFFF";
                              ctx.moveTo(cold1, cold2);
                              ctx.lineTo(coord[0] / 12.5, coord[1] / 12.5);
                              ctx.stroke();
              
                              cold1 = coord[0] / 12.5
                              cold2 = coord[1] / 12.5
                          }
                          first = false
              
                      });
                  }
                  // Zeichne Roboter
                  ctx.beginPath();
                  if (res.robot){
                      if (res.path.current_angle && typeof res.robot[0] !== "undefined" && typeof res.robot[1] !== "undefined") {
                          canvasimg = rotateRobo(img, res.path.current_angle);
                          ctx.drawImage(canvasimg, res.robot[0] / 12.5 - 15, res.robot[1] / 12.5 - 15, img.width, img.height);
                      } else {
                          ctx.drawImage(img, res.robot[0] / 12.5 - 15, res.robot[1] / 12.5 - 15, img.width, img.height);
                      }
                  }
                  // Zeichne Ladestation wenn vorhanden
                  if (res.charger){
                      if (typeof res.charger[0] !== "undefined" && typeof res.charger[1] !== "undefined") {
                          ctx.beginPath();
                          ctx.drawImage(img_charger, res.charger[0] / 12.5 - 15, res.charger[1] / 12.5 - 15);
                      }
                  }
                
              
              
                  // crop image
                  let canvas_final = createCanvas();
                  let ctx_final = canvas_final.getContext('2d');
                  var trimmed = ctx.getImageData(res.image.position.left * 4, res.image.position.top * 4, res.image.dimensions.width * 4, res.image.dimensions.height * 4);
              
                  canvas_final.height = res.image.dimensions.height * 4;
                  canvas_final.width = res.image.dimensions.width * 4;
              
                  ctx_final.putImageData(trimmed, 0, 0);
              
              
                  map = canvas_final.toDataURL();
                  last_map=  canvas_final;
                  setState("javascript.0.vis.RockroboMap", '<img src="' + canvas_final.toDataURL() + '" /style="width: auto ;height: 100%;">');
                  //log('<img src="' + canvas_final.toDataURL() + '" />');
              }
              function send(canvas){
              var buf = canvas.toBuffer();
              fs.writeFile("/opt/iobroker/vac_map.png", buf, (err) => {
                if (err) throw err;
                log('The file has been saved!');
              });
              setTimeout(function(){
                 // sendTo('telegram.0', '/opt/iobroker/vac_map.png');
                  sendTo('telegram.0', {text: '/opt/iobroker/vac_map.png', caption: 'Fertig mit saugen!'});
               }, 3000);
              
              }
              
              function rotateRobo(img, angle) {
                  var canvasimg = createCanvas(img.width, img.height);
                  var ctximg = canvasimg.getContext('2d');
                  const offset = 90;
              
                  ctximg.clearRect(0, 0, img.width, img.height);
                  ctximg.translate(img.width / 2, img.width / 2);
                  ctximg.rotate((angle + offset) * Math.PI / 180);
                  ctximg.translate(-img.width / 2, -img.width / 2);
                  ctximg.drawImage(img, 0, 0);
                  return canvasimg;
              }
              schedule("*/2 * * * * *", function () {
                  var robyState = getState(robotState).val;
              
                  if (robyState === 5 || robyState === 11 || robyState === 17) httpGetAsync("http://" + robotIp + "/api/map/latest", updateMapPage);
              });
              
              subscribe({ id: 'mihome-vacuum.0.info.state'/*Vacuum state*/, change: "ne" }, function (obj) {
              
                  if(obj.newState.val === 8 && obj.oldState.val !== 8){
                  send(last_map);
                  }
              });
              

              Guten Morgen, hab´s mal flink eingebaut. Funktioniert! Großartig.

              Vielen Dank für die schnelle Hilfe!

              1 Reply Last reply Reply Quote 0
              • First post
                Last post

              Support us

              ioBroker
              Community Adapters
              Donate

              448
              Online

              31.8k
              Users

              80.0k
              Topics

              1.3m
              Posts

              2
              4
              464
              Loading More Posts
              • Oldest to Newest
              • Newest to Oldest
              • Most Votes
              Reply
              • Reply as topic
              Log in to reply
              Community
              Impressum | Datenschutz-Bestimmungen | Nutzungsbedingungen
              The ioBroker Community 2014-2023
              logo