Navigation

    Logo
    • Register
    • Login
    • Search
    • Recent
    • Tags
    • Unread
    • Categories
    • Unreplied
    • Popular
    • GitHub
    • Docu
    • Hilfe
    1. Home
    2. Deutsch
    3. Skripten / Logik
    4. [Vorlage] todoist.com To-Do-Listen Script für VIS

    NEWS

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

    • ioBroker goes Matter ... Matter Adapter in Stable

    • Monatsrückblick - April 2025

    [Vorlage] todoist.com To-Do-Listen Script für VIS

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

      Habs gemacht wie du: googlen und ausprobieren 😉

      hier habe ich meine Infos gezogen:

      https://www.sitepoint.com/making-http-r … n-node-js/

      1 Reply Last reply Reply Quote 0
      • T
        tempestas last edited by

        ich habe hier irgendwie Fehler im Log bei meinem abgewandeltem Skript. Da scheint was noch nicht zu stimmen:

        javascript.1	2018-03-15 14:59:06.204	error	at IncomingMessage.g (events.js:292:16)
        javascript.1	2018-03-15 14:59:06.204	error	at IncomingMessage. (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:1085:12)
        javascript.1	2018-03-15 14:59:06.204	error	at Request.emit (events.js:188:7)
        javascript.1	2018-03-15 14:59:06.204	error	at emitOne (events.js:96:13)
        javascript.1	2018-03-15 14:59:06.204	error	at Request. (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:1163:10)
        javascript.1	2018-03-15 14:59:06.204	error	at Request.emit (events.js:191:7)
        javascript.1	2018-03-15 14:59:06.204	error	at emitTwo (events.js:106:13)
        javascript.1	2018-03-15 14:59:06.204	error	at Request.self.callback (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:186:22)
        javascript.1	2018-03-15 14:59:06.204	error	at Request._callback (script.js.common.Einkauf.todoist_alexa_listen2:38:29)
        javascript.1	2018-03-15 14:59:06.204	error	at Object.parse (native)
        javascript.1	2018-03-15 14:59:06.204	error	SyntaxError: Unexpected token I in JSON at position 0
        javascript.1	2018-03-15 14:59:06.203	error	uncaught exception: Unexpected token I in JSON at position 0
        

        Zeile 38 begeinnt bei mir das Task abholen. Denke, das hängt sich auf, weil nach dem Löschen keine Tasks mehr enthalten sind?

        Nachtrag: habe es mal ne ganze Weile beobachtet. Das kommt irgendwie nicht immer, obwohl ich nichts weiter geändert habe? Also auch bei Projekten ohne Tasks kommt nicht immer der Fehler. Nun bin ich planlos.

        1 Reply Last reply Reply Quote 0
        • T
          tempestas last edited by

          Ich sheitere daran, die Create Task Erklärung aus der API in mein Javascript zu "übersetzen". IRgendwie fehlt mir da der Zugang, und google hat noch nicht wirklich geholfen. Kann mir jemand den entscheidenen Tipp geben? Wie füge ich einen Task hinzu?

          wie "übersetze" ich

          $ curl "https://beta.todoist.com/API/v8/tasks" \
              -X POST \
              --data '{"content": "Appointment with Maria", "due_string": "tomorrow at 12:00", "due_lang": "en", "priority": 4}' \
              -H "Content-Type: application/json" \
              -H "X-Request-Id: $(uuidgen)" \
              -H "Authorization: Bearer $token"
          
          {
              "comment_count": 0,
              "completed": false,
              "content": "Appointment with Maria",
              "due": {
                  "date": "2016-09-01",
                  "datetime": "2016-09-01T11:00:00Z",
                  "string": "2017-07-01 12:00",
                  "timezone": "Europe/Lisbon"
              },
              "id": 123,
              "order": 20,
              "priority": 4,
              "project_id": 234,
              "url": "https://todoist.com/showTask?id=123"
          }
          
          

          in mein Javascript? Beim löschen habe ich es ja hinbekommen, aber etwas hinzufügen, da scheitere ich.

          1 Reply Last reply Reply Quote 0
          • T
            tempestas last edited by

            Also die Fehlermeldungen kommen leider weiter.

            Es bezieht sich aber augenscheinlich auf den Bereich, den ich NICHT geändert habe.

            ! ````
            host.iobrokerNUC 2018-03-16 07:55:01.368 error instance system.adapter.javascript.1 terminated with code 0 (OK)
            Caught 2018-03-16 07:55:01.368 error by controller[0]: at IncomingMessage.g (events.js:292:16)
            Caught 2018-03-16 07:55:01.367 error by controller[0]: at IncomingMessage. (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:1085:12)
            Caught 2018-03-16 07:55:01.367 error by controller[0]: at Request.emit (events.js:188:7)
            Caught 2018-03-16 07:55:01.367 error by controller[0]: at emitOne (events.js:96:13)
            Caught 2018-03-16 07:55:01.367 error by controller[0]: at Request. (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:1163:10)
            Caught 2018-03-16 07:55:01.367 error by controller[0]: at Request.emit (events.js:191:7)
            Caught 2018-03-16 07:55:01.367 error by controller[0]: at emitTwo (events.js:106:13)
            Caught 2018-03-16 07:55:01.367 error by controller[0]: at Request.self.callback (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:186:22)
            Caught 2018-03-16 07:55:01.366 error by controller[0]: at Request._callback (script.js.common.Einkauf.todoist_alexa_listen2:23:34)
            Caught 2018-03-16 07:55:01.366 error by controller[0]: at Object.parse (native)
            Caught 2018-03-16 07:55:01.364 error by controller[0]: SyntaxError: Unexpected token I in JSON at position 0
            javascript.1 2018-03-16 07:55:01.187 error at IncomingMessage.g (events.js:292:16)
            javascript.1 2018-03-16 07:55:01.187 error at IncomingMessage. (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:1085:12)
            javascript.1 2018-03-16 07:55:01.187 error at Request.emit (events.js:188:7)
            javascript.1 2018-03-16 07:55:01.187 error at emitOne (events.js:96:13)
            javascript.1 2018-03-16 07:55:01.187 error at Request. (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:1163:10)
            javascript.1 2018-03-16 07:55:01.187 error at Request.emit (events.js:191:7)
            javascript.1 2018-03-16 07:55:01.187 error at emitTwo (events.js:106:13)
            javascript.1 2018-03-16 07:55:01.187 error at Request.self.callback (/opt/iobroker/node_modules/iobroker.javascript/node_modules/request/request.js:186:22)
            javascript.1 2018-03-16 07:55:01.187 error at Request._callback (script.js.common.Einkauf.todoist_alexa_listen2:23:34)
            javascript.1 2018-03-16 07:55:01.187 error at Object.parse (native)
            javascript.1 2018-03-16 07:55:01.187 error SyntaxError: Unexpected token I in JSON at position 0
            javascript.1 2018-03-16 07:55:01.171 error uncaught exception: Unexpected token I in JSON at position 0

            
            Seltsamerweise kommt dieser Fehler auch nicht dauernd, obwohl der Schedule auf 1 Minute eingestellt ist sondern eben hier um 7:55h und davor um 6:59h? Jemand ne Idee?
            
            Nachtrag: entgegen meiner gestrigen Vermutung kommt der Fehler offenbar auch, wenn die Liste befüllt ist.
            
            Abseits vom Fehler: das hinzufügen von Extern (ohne Alexa) scheitert glaube ich i.W. am uniquie identifier, der benötigt wird. Was das nun wieder ist (außer dem rein vom Namen her offensichtlichem) und wie ich damit umgehe muss ich noch versuchen zu finden und zu verstehen
            1 Reply Last reply Reply Quote 0
            • T
              tempestas last edited by

              Guten morgen zusammen,

              ich habe das Thema weiter bearbeitet. Herausgekommen ist jetzt, dass ich die Zusammenführung meiner "manuellen" (d.h. am Tablet zu bedienenden) Liste und der todoist Alexa Liste soweit fertig habe, dass ich via Alexa Einkäufe hinzufügen kann und via Eingabe am Tablet auch. Diese Eingaben werden dann auch auf der TODOIST Liste berücksichtigt. Es funktioniert sowohl das hinzufügen als auch das löschen einzelner Dinge.

              Was noch fehlt ist jetzt die Rückkopplung in die andere Richtung, also ich füge etwas via Alexa hinzu und es wird entsprechend auf dem Tablet markiert (es soll ja nicht nur in die Liste gehen sondern der jeweilige Button muss auch true gehen). Dabei stellt sich mir neben der generellen Herausforderung aber noch das Problem, dass ich aktuell fürchte ich einen Zirkel schaffen würde, denn todoist lauscht jetzt auf Änderungen am Tablet. Wenn also das Tablet nun auf Änderungen bei Todoist lauscht, dann triggert sich beides gegenseitig. Dazu muss ich mir noch Gedanken machen.

              Aus Übersichtlichkeitsgründen habe ich derzeit drei Skripte.

              Skript 1 ist mein "manuelles Skript", visualisiert sieht das so aus:

              ! ![](</s><URL url=)<link_text text="download/file.php?id=22640&mode=view">https://forum.iobroker.net/download/file.php?id=22640&mode=view</link_text>" /> ~~das Skript:

              ! ```
              `// Einkaufsliste
              //
              // Idee und Basis: dna909
              //
              // http://forum.iobroker.net/viewtopic.php?f=21&t=8766
              //
              // abgewandelt: tempestas, Datum 6.11.2017
              //
              // status: v 1.0 - stable
              // status: v 1.1 - regex trigger über state verzögert, da sonst bei erstmaligem Ausführen ständig getriggert wird und damit Warnungen im LOG stehen;
              // - state für VIS Kategorienumschaltung eingefügt; Fehler bei PurchList.length korrigiert, die -1 entfernt.
              ! // SKRIPTEINSTELLUNGEN
              ! var debug = false;
              var force = true;
              var path = "javascript.0.Einkaufsliste."; // Pfad für generelle Variablen und States
              var pathItems = "javascript.0.Einkaufsliste.Items."; // Pfad für die zu kaufenden Gegenstände
              !
              ! // ############################
              // ########## STATES ##########
              // ############################
              ! createState(path +'regex', false, force, {
              name: "Liste",
              type: 'boolean',
              role: 'state',
              });
              ! createState(path +'Liste', "", force, { // String, in dem die Einkaufsliste gespeichert wird
              name: "Liste",
              type: 'string',
              role: 'string',
              });
              ! createState(path +'clearBtn', false, force, { // löschen Button, damit man nicht alles manuell abwählen muss
              name: 'Clear Liste',
              desc: 'Löscht die Liste',
              type: 'boolean',
              role: 'state',
              unit: ''
              });
              ! createState(path +'telegramBtn', false, force, { // sendet die Liste via Telegram
              name: 'Send Telegram',
              desc: 'Sendet Telegram Einkaufsliste',
              type: 'boolean',
              role: 'state',
              unit: ''
              });
              ! createState(path +'visCategory',0, force, { // für die verschiedenen Kategorien
              name: 'VIS Einkaufsliste',
              type: 'number',
              min: '0',
              max: '20',
              states: '0:Obst;1:Gemüse;2:Wurst/Fleisch; 3:Käse/Milchprodukte; 4: Brot/Getreide; 5: Saucen/Gewürze; 6: Getränke; 7: Drogerie 1; 8: Drogerie 2',
              desc: 'Einkaufskategorien'
              });
              ! createState(path +'visBtn', false, force, { // trigger für Anzeige der Gesamtliste in Vis
              name: 'Show Liste',
              desc: 'Zeigt die Gesamtliste in Vis',
              type: 'boolean',
              role: 'state',
              unit: ''
              });
              ! createState(path +'count',0, force, {
              name: 'VIS Einkaufsliste Anzahl Objekte',
              type: 'number',
              min: '0',
              max: '200',
              });
              ! // #########################################################
              // ARRAY MIT INHALT DER KAUFGEGENSTÄNDE
              // #########################################################
              ! var PurchList= [ "Ananas", "Apfel", "Aprikose_Pfirsich","Banane", "Birne", "Datteln", "Erdbeeren","Kirschen",
              "Kiwi", "Mandarine","Mango", "Melone", "Orange", "Waldbeeren", "Weintrauben", "Zitrone_Limette",

                                  "Chilis", "Chinakohl", "Erbsen", "Gurke","Karotten", "Kartoffeln",  "Kohlrabi", "Kürbis", "Lauch", "Mais", "Paprika",
                                  "Pilze", "Radieschen", "Tomaten", "Zucchini", "Zwiebeln", 
              
                                  "Aufschnitt", "Geflügel", "Geschnetzeltes", "Hackfleisch", "Kabanossi", "Rind", "Schwein", "Wiener","Fischstäbchen", "Garnelen", 
              
                                  "Bratfix", "Butter", "Creme-Fraiche", "Dips", "Eier", "Frischkaese", "Joghurt", "Kaese", "Kochsahne", "Kraeuterquark", "Milch",
                                  "Mozzarella", "Quark", "Sauerrahm", "Schmelzkäse", 
              
                                  "Brötchen", "Brot", "Couscous", "Haferflocken", "Kichererbsen", "Mehl", "Müsli", "Nudeln","Pizzateig", "Reis", "Sonnenblumenkerne", "Vollkornnudeln",
                                  "Mandeln",
              
                                  "Asiasauce", "Brühe","BBQ-Sauce", "Chili-Sauce", "Essig", "Ketchup", "Mayonaise", "Öl", "Paprikapulver", "Pesto", "Pfeffer", "Salz", "Tomatenmark", "Tomatensauce", "Senf", "Zucker", 
              
                                  "Saft", "Tonicwater", "Cola", "Fanta", "Sprite", "Bier", "Bier-alkoholfrei","Alsterwasser", 
                                  "Alsterwasser-alkoholfrei", "Weizenbier", "Sekt","Energydrink", "Wein", "Tee", "Kaffee", "Kaffee-entkoffiniert", 
              
                                  "Alufolie", "Backpapier", "Biomüllbeutel","Klopapier", "Küchenrolle", "Deo", "Haargel", "Rasierklingen", "Rasierschaum", "Zahnbürste", "Zahnpasta","Duschgel", "Raumduft", "Shampoo",  "Seife",  "Schwamm", "Bürste", "Spüli", 
                                  "Klarspüler", "Spülmaschinensalz", "Tabs","Badreiniger", "Glasreiniger","Fettlöser", "Bodenreiniger", "Ceranreiniger", "Weichspüler", "Waschmittel", "Fleckentferner"];
              

              !
              // ##################################################################
              // Variablen
              // ##################################################################
              ! var idRegex = path + 'regex';
              ! var idListe = path + 'Liste';
              ! var idClear = path + 'clearBtn';
              ! var idTelegram = path + 'telegramBtn';
              ! var idCount = path + 'count';
              ! var lengthListPurch=(PurchList.length); // Variablen für Auslesen wieviele Inhalte in den jeweiligen Arrays stehen

              var regexTrigger = /^javascript.0.Einkaufsliste.Items..*$/;

              ! var listeTotal = [];
              ! // ################################################################
              // ###### STATES FÜR DIE GEGENSTÄNDE IN OBJEKTEN ERZEUGEN #######
              // ################################################################
              ! function createItems(){
              var i = 0;
              for (i = 0; i < lengthListPurch; i++){
              createState(pathItems +PurchList[i], false, {
              name: 'Einkauf',
              desc: PurchList[i],
              type: 'boolean',
              role: 'state',
              unit: ''
              });

                  /*
                  createState(pathItems +(PurchList[i]) + ".Menge", 0, force, {           // hier könnte man Mengenangaben mit einbeziehen, brauche ich jedoch nicht.
                      name: 'Einkauf',
                      desc: PurchList[i],
                      type: 'number',
                      role: 'state',
                      unit: ''
                  });
               */   
              
              }
              

              ! }
              ! // #############################################################
              // TRIGGER FÜR ERZEUGUNG, LÖSCHEN UND TELEGRAM
              // #############################################################
              ! on({ id:regexTrigger,
              change: 'ne'},
              function(data) { // regex trigger, löst immer aus sobald etwas an- oder abgewählt wird
              if(getState(idRegex).val === true) {
              purchlistCreate(); // if Bedingung, damit es nicht zu warnmeldungen kommt, wenn der regex trigger ausgeführt wird aber Objekte noch nicht angelegt sind
              count();
              }

              });

              ! on(idClear, function(obj){ // trigger fürs Löschen
              if(obj.state.val === true){
              clearPurch();
              }
              });
              ! on(idTelegram, function(obj){ // trigger für Telegram
              if(obj.state.val === true){
              sendTelegram();
              }
              });
              ! // #################################################################
              // FUNCTIONS
              // #################################################################
              ! ///// Einkaufsliste erzeugen //////////////////////////////////////////////////////////////////
              ! function purchlistCreate(){
              var i = 0;
              var item;

                  listeTotal = [];
              
                  for (i = 0; i < lengthListPurch; i++){
              
                      if(getState(pathItems +PurchList[i]).val === true) {
                      item = PurchList[i];
                      listeTotal.push(item);
                      if(debug) log(item + " zur Einkaufsliste hinzugefügt ");
                      }
                  } 
              setState(idListe, listeTotal.join('
              

              ')); // Liste wird für VIS formatiert
              }

              ! function count(){
              var n = listeTotal.length;
              setState(idCount, n);
              }
              ! function clearPurch(){
              var i = 0;
              ! for (i = 0; i < lengthListPurch; i++){

                      if(getState(pathItems +PurchList[i]).val === true) {
                          setState(pathItems +PurchList[i], false);
                          setState(idListe, "");
                          setStateDelayed(idClear, false, 2000);
                      }
                  }
              

              }

              ! function sendTelegram() {

                      var telegramtext = getState(idListe).val.replace(/
              

              /g , "\n"); // telegram Liste formatieren, entfernt den Zeilenumbruch für vis und ersettz mit Zeilenumbruch Telegram

                      sendTo('telegram.0', {                                          // Nachricht per Telegram senden
                      text: "Bitte mitbringen:\n\n" + telegramtext        
                      });
              
                      setStateDelayed(idTelegram, false, 2000);
              

              }

              ! // Bei Start
              ! createItems();
              setStateDelayed(idRegex, true, 30000); // nachdem alle states angelegt sind, kann der regex trigger genutzt werden; ! Das Auslesen von todoist habe ich etwas abgewandelt und auf einen Schedule (für die verbalen Änderungen via Alexa) und einen Trigger (für die Änderungen via Tablet) abgewandelt. ! Das Skript sieht nun so aus ! >! ~~[spoiler]~~[code]// Todoist Script zur Erstellung eines HTML Codes zur Einbindung in VIS mittels "basic - String (unescaped)" Widget
              // Scriptversion: v0.2
              // Ergänzungen tempestas: Task IDs mit ausgelesen und angelegt, trigger eingebaut, Stand 17.3.2018
              // https://forum.iobroker.net/viewtopic.php?f=21&t=12563
              // https://forum.iobroker.net/viewtopic.php?f=21&p=133205#p133205
              ! //Erforderliche Angaben:
              var APItoken = "STRENG GEHEIM"; // API Token kann in den Todoist Settings auf der Webseite ausgelesen werden
              var debug = false;
              ! //Weitere Parameter ggf. anpassen:
              var instanz = 'javascript.0'; instanz = instanz + '.'; // Instanz fuer die Datenpunkte definieren
              var pfad = 'Todoist.'; // Pfad definieren
              var regexTrigger = /^javascript.0.Einkaufsliste.Items..$/; // Trigger analog der manuellen Einkaufsliste
              ! schedule("
              /1 * * * *", checkTODOIST); // Abfragezeit: jede Minute
              ! // *********************************************************
              // Scriptbeginn - Ab hier nichts mehr ändern!
              // *********************************************************
              ! function checkTODOIST() {

              var APIprojectsURL = "https://beta.todoist.com/API/v8/projects?token="+APItoken;
              var APItaskURL = "https://beta.todoist.com/API/v8/tasks?token="+APItoken;
              var request = require("request");
              var ToDoListen = []; // wird mit IDs der TO-DO Listen befuellt
              var ToDoListen_names = []; // wird mit Namen der TO-DO Listen befuellt
              
              request(APIprojectsURL, function (error, response, body) {
                  try{
                      var projects_json = JSON.parse(body);
                      for (k = 0; k < projects_json.length; k++) {
                          var projects = parseInt(projects_json[k].id);
                          var projects_name = JSON.stringify(projects_json[k].name);
                          projects_name = projects_name.replace(/\"/g, ""); //entfernt die Anfuehrungszeichen aus dem quellstring
                          ToDoListen[ToDoListen.length] = projects;
                          ToDoListen_names[ToDoListen_names.length] = projects_name;
                          createState(instanz + pfad + ToDoListen_names[k], {def: 'false',type: 'string',role: 'html', name: ToDoListen_names[k]+' HTML String'});
                          if(debug) log("Datenpunkt "+ToDoListen_names[k]+" erstellt.", "info");
                          createState(instanz + pfad + ToDoListen_names[k]+'.TaskID', {def: 'false',type: 'string',role: 'string', name: ToDoListen_names[k]+' Task IDs'});
                          if(debug) log("Datenpunkt "+ToDoListen_names[k]+'.TaskID'+" erstellt.", "info");            
                      }
                  }
                  catch(err) {log (err.message)}
              });
              
              setTimeout(function() {
                  try{
                      request(APItaskURL, function (error, response, body) {
                          var json = JSON.parse(body);
                          for (j = 0; j < ToDoListen.length; j++) {
                              var HTMLstring = "";
                              setState(instanz + pfad + ToDoListen_names[j], "leer");
                              for (i = 0; i < json.length; i++) {
                                  var Liste = parseInt(json[i].project_id);
                                  var content = JSON.stringify(json[i].content);
                                  content = content.replace(/\"/g, ""); //entfernt die Anfuehrungszeichen aus dem quellstring
                                  content = content[0].toUpperCase() + content.substring(1); // Macht den ersten Buchstaben des strings zu einem Grossbuchstaben
                                  var taskurl = JSON.stringify(json[i].url);
                                  taskurl = taskurl.replace(/\"/g, "");
                                  if (Liste == ToDoListen[j])
                                  {
                                      if(debug) log ("["+content+"] in "+ToDoListen_names[j]+" gefunden", "info");
                                      HTMLstring = HTMLstring+content+"
              

              "; // mit Link: HTMLstring = HTMLstring+"* "+content+"";
              setState(instanz + pfad + ToDoListen_names[j], HTMLstring); // "<>"+HTMLstring+"

              ! ");
              }
              }
              }
              });
              }
              catch(err) {log(err.message)}
              }, 1000);

                  setTimeout(function() {                                                 // legt IDs der einzelnen Tasks in Unterordnern an
                  request(APItaskURL, function (error, response, body) {
              
                      var json = JSON.parse(body);
              
                      for (j = 0; j < ToDoListen.length; j++) {
              
                          var HTMLstring = "";
              
                          setState(instanz + pfad + ToDoListen_names[j] +'.TaskID', "leer");
                          for (i = 0; i < json.length; i++) {
                              var Liste = parseInt(json[i].project_id);
                              var taskid = JSON.stringify(json[i].id);
                              if (Liste == ToDoListen[j])
                              {
                                  if(debug) log ("["+taskid+"] in "+ToDoListen_names[j]+" gefunden", "info");
                                  HTMLstring = HTMLstring+taskid+"
              

              "; // mit Link: HTMLstring = HTMLstring+"* "+content+"";
              setState(instanz + pfad + ToDoListen_names[j]+'.TaskID', HTMLstring); // "<>"+HTMLstring+"

              ! ");
              }
              }
              }

                   });
              }, 1000);
              

              }

              ! // ***********************************************
              // TRIGGER
              // ***********************************************
              ! on({ id:regexTrigger,
              change: 'ne'},
              function() { // regex trigger, löst immer aus sobald etwas in der manuellen Liste an- oder abgewählt wird
              ! checkTODOIST();
              }); ! Der Abgleich, sprich hinzufügen und löschen in der todoist Liste bei Änderungen am Tablet funktioniert dann mit dm dritten Skript: ! >! ~~[spoiler]~~[code]// SKRIPT FÜGT DER ALEXA EINKAUFSLISTE BEI TODOIST EINEN GEGENSTAND HINZU; WENN DIESER IN DER MANUELLEN EINKAUFSLISTE HINZUGEFÜGT WIRD BZW LÖSCHT
              ! var debug = true;
              ! var APItoken = "STRENG GEHEIM"; // API Token kann in den Todoist Settings auf der Webseite ausgelesen werden
              var idProject = nicht ganz so geheim, aber trotzdem; // Projekt-ID der Alexa-Einkaufsliste aus todoist
              var idEinkaufsIDS = "javascript.0.Todoist.Alexa-Einkaufsliste.TaskID"; // hier stehen die Task IDs
              var idEinkaufsItems = "javascript.0.Todoist.Alexa-Einkaufsliste"/Alexa-Einkaufsliste HTML String/;
              ! var regexTrigger = /^javascript.0.Einkaufsliste.Items..*$/; // Trigger analog der manuellen Einkaufsliste
              ! // **************************************************************
              // Scriptbeginn - Ab hier nichts mehr ändern!
              // **************************************************************
              ! var request = require("request");
              var uuid;
              var purchItem;
              ! // **************************************************************
              // uuid erstellen
              // **************************************************************
              ! function createUUID(){
              var dt = new Date().getTime();
              uuid = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) {
              var r = (dt + Math.random()*16)%16 | 0;
              dt = Math.floor(dt/16);
              return (c=='x' ? r :(r&0x3|0x8)).toString(16);
              });
              return uuid;
              }
              ! // ********************************************************************
              // "Aufgabe" bzw Einkaufsgegenstand zur todoist Alexa Liste hinzufügen
              // ********************************************************************
              ! function addTask(){
              ! createUUID();
              log(uuid);
              ! var options = { method: 'POST',
              url: 'https://beta.todoist.com/API/v8/tasks',
              headers:
              { 'Cache-Control': 'no-cache',
              Authorization: 'Bearer ' +APItoken,
              'X-Request-Id': uuid,
              'Content-Type': 'application/json' },
              body:
              { content: purchItem,
              project_id: +idProject,
              },
              json: true };

                  request(options, function (error, response, body) {
                    if (error) throw new Error(error);
              
                    if(debug) console.log(JSON.stringify(body));
                  });
              

              }

              ! // ********************************************************************
              // "Aufgabe" bzw Einkaufsgegenstand von todoist entfernen
              // ********************************************************************
              ! function deleteTask(){
              ! // Array der IDs aufbereiten

              var idListIDsArray =[];
              var idListIDs;
              
              idListIDs = getState(idEinkaufsIDS).val;
              if(debug)log(idListIDs);
              idListIDsArray = idListIDs.split("
              

              ");
              idListIDsArray.pop(); // aufgrund meiner uneleganten Änderungen steht als letztes immer "
              " in der Liste. Damit würde ein Element zuviel ausgegeben werden (leer). Pop kürzt den array ums letzte Element

              ! // Array der Items aufbereiten
              ! var idListArray =[];
              var idList;

              idList = getState(idEinkaufsItems).val;
              if(debug)log(idList);
              idListArray = idList.split("
              

              ");
              idListArray.pop();

              ! // Bestimmen der ID Position im Array

              var arrayPosition = idListArray.indexOf(purchItem);
              
              // Den Gegenstand über die gefundene ID löschen
              

              ! var deleteURL = "https://beta.todoist.com/API/v8/tasks/"+idListIDsArray[arrayPosition]+"?token="+APItoken;
              if(debug) log("Delete URL ist "+deleteURL);

                  request({
                      uri: deleteURL,
                      method: "DELETE",
                      timeout: 5000,
                      }, 
                          function(error, response, body) {
                          if(debug) console.log(body);
                      });    
              

              }

              ! // **************************************************************
              // Trigger und Wertübergabe
              // **************************************************************
              ! on({ id:regexTrigger, change: 'ne'}, // regex trigger, löst immer aus sobald etwas an- oder abgewählt wird
              function(data) {
              var temporary;
              temporary = data.id;
              purchItem = temporary.substring(33); // Pfad bis zum Item ist 33 Stellen lang

                          if(data.state.val) {                                // wenn das item auf der manuellen Liste hinzugefügt wurde (state ist true) dann...
                              if (debug) log("Es wird hinzugefügt: "+purchItem);
                              setTimeout(addTask, 500); }
                          else {                                              // sonst wenn wenn das item auf der manuellen Liste entfernt wurde dann...
                              if (debug) log("Es wird entfernt: "+purchItem);
                              setTimeout(deleteTask, 500); 
                          }    
                  });
              

              ! [/code]`[/spoiler]
              ! Nun fehlen noch #Abgleiche beim Start und eben "verbal to manual" Angleichungen.
              ! Ich hoffe, der eine oder andere findet es vielleicht nützlich
              ! VG und schönes Wochenende[/i][/i][/i][/i][/i][/code][/spoiler][/i][/i][/i][/i][/i][/i][/i][/i]

              1 Reply Last reply Reply Quote 0
              • haselchen
                haselchen Most Active last edited by

                Erstmal nen fettes Danke für Deine Arbeit,Mühen und Ausdauer.

                Diese ToDoList mit Alexa für VIS ist so das letzte I Tüpfelchen für die Tabletvisualisierung für mich.

                Nun kommt das aber 🙂

                Als blutiger Anfänger würde ich Dich um eine Anleitung bitten, wie welches Skript mit welchen Infos wo eingebunden werden muss.

                Ich habe von der Reihenfolge etc. leider null Plan.

                Ich möchte halt Alexa meinen Einkaufswunsch sagen, der soll dann auf dem Tablet erscheinen und nach Bedarf mit Telegram an meine Frau schicken.

                Wenn dann noch das löschen der Produkte per Sprache geht, lauf ich nackt durch den Ort 😄

                P.S.: Wenn ich könnte, würde ich Dir 100 Danke geben.

                Kleine Nebenfrage noch: Funktioneren auch Mengenangaben? 3kg Äpfel oder 1Liter Milch?

                1 Reply Last reply Reply Quote 0
                • T
                  tempestas last edited by

                  Hallo haselchen,

                  also die erste Frage ist: willst du NUR via Spracheingabe deine Einkaufsliste erstellen? Dann reicht ein Skript.

                  Folgende Einschränkungen gibt es, die von uns hier auch nicht behoben werden können, sondern an der Alexa Integration liegen:

                  1. Alles, was du in einem Atemzug sagst, wird als ein String aufgeschrieben. D.h: Wenn du "Setze Äpfel, Bier und Weißbrot auf meine Einkaufsliste" sagst, dann erscheint das genau so auf deiner Liste. Genau genommen noch mit Rechtschreibfehlern, nämlich "Äpfel, bier und weißbrot". Dieser String ist dann eine "Einheit". Wenns nicht stört ist das ok, ich wollte dich nur drauf hinweisen.

                  2. Ich habe es gerade getestet: man kann NICHT via Sprachbefehl Dinge löschen. Alexa sagt dann "gehe in die Alexa App". Das nur als Hinweis. Zum löschen der gesamten Liste gibt es aber das von mir oben genannte kleine Skript. Damit könntest du dann am Tablet zumindest alles löschen, ohne in die Alexa App reinzugehen.

                  3. Du kannst "Setze 3kg Äpfel auf meine Einkaufsliste" sagen. Allerdings sieht das in der Liste dann so aus: "Drei kilogramm äpfel".

                  Um todoist mit Alexa zu verknüpfen, schaue hier:

                  https://support.todoist.com/hc/en-us/ar … azon-Echo-

                  Als Skript nutzt du das zweite aus meinem obigen Posting.

                  Zum Thema löschen und integration in vis melde ich mich morgen, muss nun los.

                  1 Reply Last reply Reply Quote 0
                  • haselchen
                    haselchen Most Active last edited by

                    Vielen Dank für die ersten Antworten 🙂

                    1 Reply Last reply Reply Quote 0
                    • T
                      tempestas last edited by

                      Hallo haselchen,

                      hier also die Step by Step Anleitung:

                      1. bei todoist anmelden (eigener account, facebook, google)

                      2. Alexa mit todoist verbinden https://blog.todoist.com/de/2017/03/01/ … tegration/

                      3. deinen API Key auslesen auf der todoist Homepage (unter Einstellungen findest du den) und im Skript eintragen

                      4. Dieses Skript ist das Hauptskript zum Visualisieren der todoist Liste

                      ! ```
                      `// Todoist Script zur Erstellung eines HTML Codes zur Einbindung in VIS mittels "basic - String (unescaped)" Widget
                      // Scriptversion: v0.2
                      // Ergänzungen tempestas, Stand 17.3.2018
                      // https://forum.iobroker.net/viewtopic.php?f=21&t=12563
                      // https://forum.iobroker.net/viewtopic.php?f=21&p=133205#p133205
                      ! //Erforderliche Angaben:
                      var APItoken = "xxxxxxxxxxxxxxxxxxx"; // API Token kann in den Todoist Settings auf der Webseite ausgelesen werden
                      var debug = false;
                      var idEinkaufsliste= "javascript.0.Todoist.Alexa-Einkaufsliste";
                      var force = true;
                      ! //Weitere Parameter ggf. anpassen:
                      var instanz = 'javascript.0'; instanz = instanz + '.'; // Instanz fuer die Datenpunkte definieren
                      var pfad = 'Todoist.'; // Pfad definieren
                      // var regexTrigger = /^javascript.0.Einkaufsliste.Items..$/; // Trigger analog der manuellen Einkaufsliste, brauchst du nicht, deaktiviert
                      ! schedule("
                      /1 * * * *", checkTODOIST); // Abfragezeit: jede Minute
                      ! // *********************************************************
                      // Scriptbeginn - Ab hier nichts mehr ändern!
                      // *********************************************************
                      ! createState(instanz + pfad +'telegramBtn', false, force, { // sendet die Liste via Telegram
                      name: 'Send Telegram',
                      desc: 'Sendet Telegram Einkaufsliste',
                      type: 'boolean',
                      role: 'state',
                      unit: ''
                      });
                      ! var idTelegram = instanz + pfad +"telegramBtn";
                      ! function sendTelegram() {

                              var telegramtext = getState(idEinkaufsliste).val.replace(/
                      

                      /g , "\n"); // telegram Liste formatieren, entfernt den Zeilenumbruch für vis und ersettz mit Zeilenumbruch Telegram

                              sendTo('telegram.0', {                                          // Nachricht per Telegram senden
                              text: "Bitte mitbringen:\n\n" + telegramtext        
                              });
                      
                              setStateDelayed(idTelegram, false, 2000);
                      

                      }

                      ! // Trigger Telegram
                      ! on(idTelegram, function(obj){ // trigger für Telegram
                      if(obj.state.val === true){
                      sendTelegram();
                      }
                      });
                      ! function checkTODOIST() {

                      var APIprojectsURL = "https://beta.todoist.com/API/v8/projects?token="+APItoken;
                      var APItaskURL = "https://beta.todoist.com/API/v8/tasks?token="+APItoken;
                      var request = require("request");
                      var ToDoListen = []; // wird mit IDs der TO-DO Listen befuellt
                      var ToDoListen_names = []; // wird mit Namen der TO-DO Listen befuellt
                      
                      request(APIprojectsURL, function (error, response, body) {
                          try{
                              var projects_json = JSON.parse(body);
                              for (k = 0; k < projects_json.length; k++) {
                                  var projects = parseInt(projects_json[k].id);
                                  var projects_name = JSON.stringify(projects_json[k].name);
                                  projects_name = projects_name.replace(/\"/g, ""); //entfernt die Anfuehrungszeichen aus dem quellstring
                                  ToDoListen[ToDoListen.length] = projects;
                                  ToDoListen_names[ToDoListen_names.length] = projects_name;
                                  createState(instanz + pfad + ToDoListen_names[k], {def: 'false',type: 'string',role: 'html', name: ToDoListen_names[k]+' HTML String'});
                                  if(debug) log("Datenpunkt "+ToDoListen_names[k]+" erstellt.", "info");
                                  createState(instanz + pfad + ToDoListen_names[k]+'.TaskID', {def: 'false',type: 'string',role: 'string', name: ToDoListen_names[k]+' Task IDs'});
                                  if(debug) log("Datenpunkt "+ToDoListen_names[k]+'.TaskID'+" erstellt.", "info");            
                              }
                          }
                          catch(err) {log (err.message)}
                      });
                      
                      setTimeout(function() {
                          try{
                              request(APItaskURL, function (error, response, body) {
                                  var json = JSON.parse(body);
                                  for (j = 0; j < ToDoListen.length; j++) {
                                      var HTMLstring = "";
                                      setState(instanz + pfad + ToDoListen_names[j], "leer");
                                      for (i = 0; i < json.length; i++) {
                                          var Liste = parseInt(json[i].project_id);
                                          var content = JSON.stringify(json[i].content);
                                          content = content.replace(/\"/g, ""); //entfernt die Anfuehrungszeichen aus dem quellstring
                                          content = content[0].toUpperCase() + content.substring(1); // Macht den ersten Buchstaben des strings zu einem Grossbuchstaben
                                          var taskurl = JSON.stringify(json[i].url);
                                          taskurl = taskurl.replace(/\"/g, "");
                                          if (Liste == ToDoListen[j])
                                          {
                                              if(debug) log ("["+content+"] in "+ToDoListen_names[j]+" gefunden", "info");
                                              HTMLstring = HTMLstring+content+"
                      

                      "; // mit Link: HTMLstring = HTMLstring+"* "+content+"";
                      setState(instanz + pfad + ToDoListen_names[j], HTMLstring); // "<>"+HTMLstring+"

                      ! ");
                      }
                      }
                      }
                      });
                      }
                      catch(err) {log(err.message)}
                      }, 1000);

                          setTimeout(function() {                                                 // legt IDs der einzelnen Tasks in Unterordnern an
                          request(APItaskURL, function (error, response, body) {
                      
                              var json = JSON.parse(body);
                      
                              for (j = 0; j < ToDoListen.length; j++) {
                      
                                  var HTMLstring = "";
                      
                                  setState(instanz + pfad + ToDoListen_names[j] +'.TaskID', "leer");
                                  for (i = 0; i < json.length; i++) {
                                      var Liste = parseInt(json[i].project_id);
                                      var taskid = JSON.stringify(json[i].id);
                                      if (Liste == ToDoListen[j])
                                      {
                                          if(debug) log ("["+taskid+"] in "+ToDoListen_names[j]+" gefunden", "info");
                                          HTMLstring = HTMLstring+taskid+"
                      

                      "; // mit Link: HTMLstring = HTMLstring+"* "+content+"";
                      setState(instanz + pfad + ToDoListen_names[j]+'.TaskID', HTMLstring); // "<>"+HTMLstring+"

                      ! ");
                      }
                      }
                      }

                           });
                      }, 1000);
                      

                      }`
                      kopierst du in eine Javascript Instanz in ein neues Skript,speicherst und startest via Play Button. Du wirst im Log gelbe Warnmeldungen sehen, was daher kommt, dass die states noch nicht angelegt sind. Das Skript nochmal stoppen, nochmal starten.

                      ! 5) mit diesem Widget
                      ! [spoiler]~~[code]~~[{"tpl":"tplValueStringRaw","data":{"oid":"javascript.0.Todoist.Alexa-Einkaufsliste","g_fixed":true,"g_visibility":false,"g_css_font_text":true,"g_css_background":true,"g_css_shadow_padding":false,"g_css_border":false,"g_gestures":false,"g_signals":false,"g_last_change":false,"visibility-cond":"==","visibility-val":1,"visibility-groups-action":"hide","signals-cond-0":"==","signals-val-0":true,"signals-icon-0":"/vis/signals/lowbattery.png","signals-icon-size-0":0,"signals-blink-0":false,"signals-horz-0":0,"signals-vert-0":0,"signals-hide-edit-0":false,"signals-cond-1":"==","signals-val-1":true,"signals-icon-1":"/vis/signals/lowbattery.png","signals-icon-size-1":0,"signals-blink-1":false,"signals-horz-1":0,"signals-vert-1":0,"signals-hide-edit-1":false,"signals-cond-2":"==","signals-val-2":true,"signals-icon-2":"/vis/signals/lowbattery.png","signals-icon-size-2":0,"signals-blink-2":false,"signals-horz-2":0,"signals-vert-2":0,"signals-hide-edit-2":false,"lc-type":"last-change","lc-is-interval":true,"lc-is-moment":false,"lc-format":"","lc-position-vert":"top","lc-position-horz":"right","lc-offset-vert":0,"lc-offset-horz":0,"lc-font-size":"12px","lc-font-family":"","lc-font-style":"","lc-bkg-color":"","lc-color":"","lc-border-width":"0","lc-border-style":"","lc-border-color":"","lc-border-radius":10,"lc-zindex":0,"html_prepend":"**Einkaufsliste:** **","name":"Einkaufsliste"},"style":{"left":"492px","top":"81px","width":"260px","height":"350px","color":"rgba(250,250,250,1)","background":"rgba(150,150,150,0.5)","z-index":"2"},"widgetSet":"basic"}][/code][/spoiler]
                      wird deine Einkaufsliste angezeigt (in vis auf widget importieren und diesen code eintragen, copy & paste)
                      ! 6) ein neues Skript anlegen mit diesem Skript zum Löschen
                      ! >! [spoiler]`[code]//Script zum Löschen aller todoist Einträge via Vis
                      // Scriptversion: v0.1
                      // https://forum.iobroker.net/viewtopic.php?f=21&t=12563
                      // https://forum.iobroker.net/viewtopic.php?f=21&p=133205#p133205
                      ! //Erforderliche Angaben:
                      var APItoken = "XXXXXXXXXXXXXXXXXXXX"; // API Token kann in den Todoist Settings auf der Webseite ausgelesen werden
                      var idEinkaufsliste = "javascript.0.Todoist.Alexa-Einkaufsliste.TaskID"/Alexa-Einkaufsliste HTML String/;
                      var debug = false;
                      var instanz = 'javascript.0'; instanz = instanz + '.'; // Instanz fuer die Datenpunkte definieren
                      var pfad = 'Todoist.'; // Pfad definieren
                      var force = true;
                      ! // Scriptbeginn - Ab hier nichts mehr ändern!
                      ! createState(instanz + pfad +'clearBtn', false, force, { // löschen Button, damit man nicht alles manuell abwählen muss
                      name: 'Clear Liste',
                      desc: 'Löscht die Liste',
                      type: 'boolean',
                      role: 'state',
                      unit: ''
                      });
                      ! var idClear = instanz + pfad + 'clearBtn';
                      ! var request = require("request");
                      !
                      function deleteTask(){
                      ! var idListArray =[];
                      var idList;

                      idList = getState(idEinkaufsliste).val;
                      if(debug)log(idList);
                      idListArray = idList.split("
                      

                      ");
                      idListArray.pop(); // aufgrund meiner uneleganten Änderungen steht als letztes immer "
                      " in der Liste. Damit würde ein Element zuviel ausgegeben werden (leer). Pop kürzt den array ums letzte Element

                      if(debug)log("idList Array ist " +idListArray);
                      if(debug)log("idList Array Länge ist "+idListArray.length);
                      
                      for (i = 0; i< idListArray.length; i++){
                      
                          if(debug) log("Durchgang "+i);
                      
                          var deleteURL = "https://beta.todoist.com/API/v8/tasks/"+idListArray[i]+"?token="+APItoken;
                          if(debug) log("Delete URL ist "+deleteURL);
                      
                          request({
                              uri: deleteURL,
                              method: "DELETE",
                              timeout: 5000,
                              }, 
                                  function(error, response, body) {
                                  if(debug) console.log(body);
                              });    
                          }
                      

                      }

                      ! // Löschen der Liste
                      ! on(idClear, function(obj){ // trigger fürs Löschen
                      if(obj.state.val === true){
                      deleteTask();
                      setStateDelayed(idClear, false, 1000);
                      }
                      }); ! 7) button in vis erstellen zum Löschen: ! >! ~~[spoiler]~~[code][{"tpl":"tplMetroTileToggle","data":{"oid":"javascript.0.Todoist.clearBtn","g_fixed":true,"g_visibility":false,"g_css_font_text":false,"g_css_background":false,"g_css_shadow_padding":false,"g_css_border":false,"g_gestures":false,"g_signals":false,"g_last_change":false,"visibility-cond":"==","visibility-val":1,"visibility-groups-action":"hide","hover":"true","transform":"true","bg_class_false":"bg-crimson","bg_class_true":"bg-crimson","icon_class_false":"icon-cancel-2","icon_class_true":"icon-cancel-2","signals-cond-0":"==","signals-val-0":true,"signals-icon-0":"/vis/signals/lowbattery.png","signals-icon-size-0":0,"signals-blink-0":false,"signals-horz-0":0,"signals-vert-0":0,"signals-hide-edit-0":false,"signals-cond-1":"==","signals-val-1":true,"signals-icon-1":"/vis/signals/lowbattery.png","signals-icon-size-1":0,"signals-blink-1":false,"signals-horz-1":0,"signals-vert-1":0,"signals-hide-edit-1":false,"signals-cond-2":"==","signals-val-2":true,"signals-icon-2":"/vis/signals/lowbattery.png","signals-icon-size-2":0,"signals-blink-2":false,"signals-horz-2":0,"signals-vert-2":0,"signals-hide-edit-2":false,"lc-type":"last-change","lc-is-interval":true,"lc-is-moment":false,"lc-format":"","lc-position-vert":"top","lc-position-horz":"right","lc-offset-vert":0,"lc-offset-horz":0,"lc-font-size":"12px","lc-font-family":"","lc-font-style":"","lc-bkg-color":"","lc-color":"","lc-border-width":"0","lc-border-style":"","lc-border-color":"","lc-border-radius":10,"lc-zindex":0,"g_badge":true,"icon_badge_false":"","icon_badge_true":"","badge_bg_class_false":"","badge_bg_class_true":"","brand_bg_class_false":"","brand_bg_class_true":"","badge_top":"0","badge_left":"0","g_ccontrol":true,"label_false":"Liste löschen","label_true":"Liste löschen","icon_false":"","icon_true":"","icon_width":"75","icon_height":"75","icon_top":"15","icon_left":"28","name":"Löschen_btn"},"style":{"left":"1123px","top":"1px","width":"150px","height":"150px","z-index":"5"},"widgetSet":"metro"}][/code][/spoiler] ! 8) button zum Telegram senden: ! >! ~~[spoiler]~~[code][{"tpl":"tplMetroTileToggle","data":{"oid":"javascript.0.Todoist.telegramBtn","g_fixed":true,"g_visibility":false,"g_css_font_text":false,"g_css_background":false,"g_css_shadow_padding":false,"g_css_border":false,"g_gestures":false,"g_signals":false,"g_last_change":false,"visibility-cond":"==","visibility-val":1,"visibility-groups-action":"hide","hover":"true","transform":"true","bg_class_false":"bg-dark","bg_class_true":"bg-steel","icon_class_false":"","icon_class_true":"","signals-cond-0":"==","signals-val-0":true,"signals-icon-0":"/vis/signals/lowbattery.png","signals-icon-size-0":0,"signals-blink-0":false,"signals-horz-0":0,"signals-vert-0":0,"signals-hide-edit-0":false,"signals-cond-1":"==","signals-val-1":true,"signals-icon-1":"/vis/signals/lowbattery.png","signals-icon-size-1":0,"signals-blink-1":false,"signals-horz-1":0,"signals-vert-1":0,"signals-hide-edit-1":false,"signals-cond-2":"==","signals-val-2":true,"signals-icon-2":"/vis/signals/lowbattery.png","signals-icon-size-2":0,"signals-blink-2":false,"signals-horz-2":0,"signals-vert-2":0,"signals-hide-edit-2":false,"lc-type":"last-change","lc-is-interval":true,"lc-is-moment":false,"lc-format":"","lc-position-vert":"top","lc-position-horz":"right","lc-offset-vert":0,"lc-offset-horz":0,"lc-font-size":"12px","lc-font-family":"","lc-font-style":"","lc-bkg-color":"","lc-color":"","lc-border-width":"0","lc-border-style":"","lc-border-color":"","lc-border-radius":10,"lc-zindex":0,"g_badge":true,"icon_badge_false":"","icon_badge_true":"","badge_bg_class_false":"","badge_bg_class_true":"","brand_bg_class_false":"","brand_bg_class_true":"","badge_top":"0","badge_left":"0","g_ccontrol":true,"label_false":"Sende Liste","label_true":"Sende Liste","icon_false":"/vis.0/main/img/telegram_transparent_neu.png","icon_true":"/vis.0/main/img/telegram_transparent_neu.png","icon_width":"75","icon_height":"67","icon_top":"33","icon_left":"28","name":"Telegram_btn"},"style":{"left":"1122px","top":"156px","width":"150px","height":"150px","z-index":"5"},"widgetSet":"metro"}][/code][/spoiler] ! 9) und als goodie: mit diesem Skript kannst du die Einkaufsliste nicht nur senden, sondern aktiv via Telegram auf dienem Handy abrufen. Dazu musst du dienen Bot mit "Einkauf" anschreiben und er wird antworten ! >! ~~[spoiler]~~~~[code]~~var idListe = "javascript.0.Todoist.Alexa-Einkaufsliste"/Liste/; // ggf anpassen
                      ! on({id: 'telegram.0.communicate.request', change: 'any'}, function (obj) {
                      var stateval = getState('telegram.0.communicate.request').val; // Statevalue in Variable schreiben
                      var benutzer = stateval.substring(1,stateval.indexOf("]")); // Benutzer aus Statevalue extrahieren
                      var befehl = stateval.substring(stateval.indexOf("]")+1,stateval.length); // Befehl/Text aus Statevalue extrahieren
                      ! if (befehl.search(/(?=.*(\Einkauf\b)).+/ig) != -1) {

                          var einkaufsliste; 
                          if(getState(idListe).val === "leer") {einkaufsliste = "Es muss nichts eingekauft werden";}
                      
                              else {  einkaufsliste = "Bitte mitbringen:\n\n" + getState(idListe).val.replace(/
                      

                      /g , "\n");}

                          sendTo('telegram', {
                              user: benutzer,
                              text: einkaufsliste
                              });
                          }    
                      

                      });
                      [/code]`[/spoiler]

                      ! die buttons sind übrigens nur beispiele, aber sie sind schonmal korrekt verlinkt, so dass du testen kannst[/i][/code][/spoiler][/i][/i][/i][/i][/i]**

                      1 Reply Last reply Reply Quote 0
                      • T
                        tempestas last edited by

                        hallo haselchen,

                        hat soweit alles funktioniert?

                        1 Reply Last reply Reply Quote 0
                        • D
                          dali3 last edited by

                          Ich habe es getestet,

                          bei mir taucht jedoch folgender Fehler auf:

                          ! ````
                          javascript.0 2018-03-20 11:23:03.403 error at ContextifyScript.Script.runInContext (vm.js:35:29)
                          javascript.0 2018-03-20 11:23:03.402 error at script.js.Todo.SkriptVisToDoIst:16:48
                          javascript.0 2018-03-20 11:23:03.402 error ReferenceError: force is not defined
                          javascript.0 2018-03-20 11:23:03.401 error ^
                          javascript.0 2018-03-20 11:23:03.401 error createState(instanz + pfad +'clearBtn', false, force, { // löschen Button, damit man nicht alles manuell abwählen muss
                          javascript.0 2018-03-20 11:23:03.400 error script.js.Todo.SkriptVisToDoIst: script.js.Todo.SkriptVisToDoIst:16
                          javascript.0 2018-03-20 11:23:03.387 info Start javascript script.js.Todo.SkriptVisToDoIst
                          javascript.0 2018-03-20 11:19:18.527 error at ContextifyScript.Script.runInContext (vm.js:35:29)
                          javascript.0 2018-03-20 11:19:18.526 error at script.js.Todo.SkriptToDoIst:24:51
                          javascript.0 2018-03-20 11:19:18.526 error ReferenceError: force is not defined
                          javascript.0 2018-03-20 11:19:18.525 error ^
                          javascript.0 2018-03-20 11:19:18.525 error createState(instanz + pfad +'telegramBtn', false, force,
                          javascript.0 2018-03-20 11:19:18.524 error script.js.Todo.SkriptToDoIst: script.js.Todo.SkriptToDoIst:24
                          javascript.0 2018-03-20 11:19:18.506 info Start javascript script.js.Todo.SkriptToDoIst

                          
                          jedoch arbeite ich via "Fernzugriff", müsste es zu Hause heute Abend nochmals testen.
                          
                          ~~@tempestas:~~ 
                          
                          > hallo haselchen,
                          > 
                          > hat soweit alles funktioniert? ` 
                          1 Reply Last reply Reply Quote 0
                          • T
                            tempestas last edited by

                            Danke für den Hinweis.

                            Habe ich übersehen; ich habe in das Skript ein paar Dinge schnell eingebaut für haselchen, dabei ist wohl was durchgerutscht.

                            Passe ich gleich an.

                            Bitte im "Hauptskript" und im Löschen-Skript einfach das hier ergänzen bei den "erforderlichen Eingaben". Habs oben auch korrigiert.

                            var force = true;
                            

                            Fernzugriff ist für das Skript natürlich nicht sinnvoll, da ich für haselchen ja nur die Version für Spracheingabe dargestellt habe. Da müsstest du wohl schon sehr, sehr laut rufen 😉

                            1 Reply Last reply Reply Quote 0
                            • D
                              dali3 last edited by

                              Gibt es auch eine Google Home Version? 😉

                              Nun läuft es! Danke dir!

                              Einen Punkt habe ich noch..

                              Falls ich via Telegram "Einkauf" die Liste anfordere, erhalte ich zuerst "Ich verstehe "Einkauf" nicht…

                              gefolgt von der Liste "Bitte mitbringen..."

                              Wie krieg ich das "Ich vestehe ... nicht" weg?

                              Ergänzung: Ich versuche nun via IFTTT google Home mit einzubinden!

                              1 Reply Last reply Reply Quote 0
                              • T
                                tempestas last edited by

                                Alexa sagt das? Ok, ich habs noch nicht in der Kombi getestet, das wäre ein unerwartete Nebenreaktion, deren Grund mir noch unklar ist.

                                1 Reply Last reply Reply Quote 0
                                • D
                                  dali3 last edited by

                                  Nein, nicht Alexa.

                                  Wenn ich im Telegram am Bot die Nachricht "Einkauf" sende…

                                  erhalte ich zuerst "Ich verstehe "Einkauf" nicht" und anschliessend die Einkaufsliste.

                                  @tempestas:

                                  Alexa sagt das? Ok, ich habs noch nicht in der Kombi getestet, das wäre ein unerwartete Nebenreaktion, deren Grund mir noch unklar ist. `

                                  1 Reply Last reply Reply Quote 0
                                  • T
                                    tempestas last edited by

                                    ok, danke.

                                    Moment; steht das "ich verstehe nicht…" mit auf der Einkaufsliste? mir ist unklar, wieso da überhaupt sowas kommen kann, denn der Bot redet ja gar nicht alleine.

                                    Bei mir passiert das auch nicht.

                                    Zu google home: das müsste genau so funktionieren, wenn sich google home mit todoist verbinden lässt.

                                    Die API bleibt ja gleich, und die Integration google bzw alexa mit todoist erfolgt ja unabhängig vom ioBroker

                                    1 Reply Last reply Reply Quote 0
                                    • D
                                      dali3 last edited by

                                      Bilder sagen mehr als 1000 Worte.

                                      Jedoch habe ich glaubs auch ein anderes Problem 🙂

                                      siehe /start hmm
                                      4746_20-03-2018_13-17-38.jpg

                                      1 Reply Last reply Reply Quote 0
                                      • T
                                        tempestas last edited by

                                        ok, das liegt dann aber irgendwie an Telegram bzw dem Bot. Wenn er auch schon bei "/start" rummeckert ist das wohl eher extern induziert und hat nichts mit dem Skript zu tun.

                                        Aber die "Leer"-Meldung hat zu einer kleinen Anpassung geführt, die ich gleich mal einbaue. Der State ist ja nie leer, sondern wenn er "leer" sein soll, eben mit "leer" beschrieben.

                                        Habs im Skript angepasst

                                        1 Reply Last reply Reply Quote 0
                                        • E
                                          eXTreMe last edited by

                                          blöde frage: aber warum nutzt ihr nicht einfach die offizielle todoist app zur anzeige der listen statt es euch aufwendig über telegram nach eingabe eines keywords zuzusenden?

                                          1 Reply Last reply Reply Quote 0
                                          • T
                                            tempestas last edited by

                                            Ist ja die Frage, ob das aufwendig ist.

                                            Ich persönlich habe bei mir alles so gemacht, dass ich möglichst wenig verschiedene Apps brauche. Über Telegram kommuniziert mein Haus mit mir, daher habe ich die Einkaufsliste da mit eingebaut.

                                            Ich nutze ja nicht eine Integrationsplattform wie ioBroker, um dann auf meinem Handy doch wieder alles auf Einzelbasis mit zig Apps zu nutzen. Ich könnte ja auch meinen Neato D5 mit der passenden App steuern oder meine Hue Lampen oder meine Homematic Sachen mit cloudmatic usw usf.

                                            Aber natürlich kann man es anders machen. Der User wollte ja nur gerne sich die Liste schicken können und quasi als "on top" gab es dann eben, falls man das schicken vergessen hat, noch die Pull Option.

                                            Man kann natürlich auch einfach eine VIS mobile view erstellen (so hab ich es).

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

                                            Support us

                                            ioBroker
                                            Community Adapters
                                            Donate

                                            781
                                            Online

                                            31.7k
                                            Users

                                            79.7k
                                            Topics

                                            1.3m
                                            Posts

                                            28
                                            130
                                            18329
                                            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