Skip to content
  • Home
  • Aktuell
  • Tags
  • 0 Ungelesen 0
  • Kategorien
  • Unreplied
  • Beliebt
  • GitHub
  • Docu
  • Hilfe
Skins
  • Light
  • Brite
  • Cerulean
  • Cosmo
  • Flatly
  • Journal
  • Litera
  • Lumen
  • Lux
  • Materia
  • Minty
  • Morph
  • Pulse
  • Sandstone
  • Simplex
  • Sketchy
  • Spacelab
  • United
  • Yeti
  • Zephyr
  • Dark
  • Cyborg
  • Darkly
  • Quartz
  • Slate
  • Solar
  • Superhero
  • Vapor

  • Standard: (Kein Skin)
  • Kein Skin
Einklappen
ioBroker Logo

Community Forum

donate donate
  1. ioBroker Community Home
  2. Deutsch
  3. Skripten / Logik
  4. JavaScript
  5. [gelöst] Mi Robot Map per Telegram

NEWS

  • Neuer Blogbeitrag: Monatsrückblick - Dezember 2025 🎄
    BluefoxB
    Bluefox
    4
    1
    27

  • Weihnachtsangebot 2025! 🎄
    BluefoxB
    Bluefox
    24
    1
    1.4k

  • UPDATE 31.10.: Amazon Alexa - ioBroker Skill läuft aus ?
    apollon77A
    apollon77
    48
    3
    9.3k

[gelöst] Mi Robot Map per Telegram

Geplant Angeheftet Gesperrt Verschoben Gelöst JavaScript
4 Beiträge 2 Kommentatoren 557 Aufrufe 3 Watching
  • Älteste zuerst
  • Neuste zuerst
  • Meiste Stimmen
Antworten
  • In einem neuen Thema antworten
Anmelden zum Antworten
Dieses Thema wurde gelöscht. Nur Nutzer mit entsprechenden Rechten können es sehen.
  • AxelF1977A Offline
    AxelF1977A Offline
    AxelF1977
    schrieb am zuletzt editiert von AxelF1977
    #1

    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

    ASROCK Deskmini Intel I3 8100 16GB mit Proxmox VM ioBroker VM DIYHue| CCU piVCCU + FHEM auf Raspberry | Maria DB mit Grafana und Prometheus auf Tinker Board

    1 Antwort Letzte Antwort
    0
    • MeistertrM Offline
      MeistertrM Offline
      Meistertr
      Developer
      schrieb am zuletzt editiert von
      #2

      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);
          }
      });
      
      AxelF1977A 1 Antwort Letzte Antwort
      0
      • MeistertrM Offline
        MeistertrM Offline
        Meistertr
        Developer
        schrieb am zuletzt editiert von
        #3

        Habs auch hochgeladen

        1 Antwort Letzte Antwort
        0
        • MeistertrM Meistertr

          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);
              }
          });
          
          AxelF1977A Offline
          AxelF1977A Offline
          AxelF1977
          schrieb am zuletzt editiert von
          #4

          @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!

          ASROCK Deskmini Intel I3 8100 16GB mit Proxmox VM ioBroker VM DIYHue| CCU piVCCU + FHEM auf Raspberry | Maria DB mit Grafana und Prometheus auf Tinker Board

          1 Antwort Letzte Antwort
          0
          Antworten
          • In einem neuen Thema antworten
          Anmelden zum Antworten
          • Älteste zuerst
          • Neuste zuerst
          • Meiste Stimmen


          Support us

          ioBroker
          Community Adapters
          Donate

          666

          Online

          32.5k

          Benutzer

          81.7k

          Themen

          1.3m

          Beiträge
          Community
          Impressum | Datenschutz-Bestimmungen | Nutzungsbedingungen | Einwilligungseinstellungen
          ioBroker Community 2014-2025
          logo
          • Anmelden

          • Du hast noch kein Konto? Registrieren

          • Anmelden oder registrieren, um zu suchen
          • Erster Beitrag
            Letzter Beitrag
          0
          • Home
          • Aktuell
          • Tags
          • Ungelesen 0
          • Kategorien
          • Unreplied
          • Beliebt
          • GitHub
          • Docu
          • Hilfe