Navigation

    Logo
    • Register
    • Login
    • Search
    • Recent
    • Tags
    • Unread
    • Categories
    • Unreplied
    • Popular
    • GitHub
    • Docu
    • Hilfe
    1. Home
    2. Deutsch
    3. Skripten / Logik
    4. [gelöst] in SayIt Wetter und Geburtstagsscript -> Müllkalender mit einbinden

    NEWS

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

    • ioBroker goes Matter ... Matter Adapter in Stable

    • Monatsrückblick - April 2025

    [gelöst] in SayIt Wetter und Geburtstagsscript -> Müllkalender mit einbinden

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

      @skorpil:

      Guten Tag, liebe Forumnesen :lol:

      die letzten Tage habe ich daran gearbeitet, mir auch "zukünftige" Müllereignisse ansagen zu lassen. Zunächst bin ich gescheitert und habe unseren Guru pix um Hilfe gebeten. Von dem kam dann folgende PN, die ich hier in Teilen veröffentliche

      ` > Ein Skript pro Thema (Müll, Geburtstag, etc.).

      Ergebnis in Datenpunkte speichern.

      Skript kann einmal täglich aufgerufen werden, meinetwegen auch stündlich.

      Dann in einem anderen Skript die Datenpunkte einlesen und zu einer Sprachausgabe zusammen basteln (schedule-, Tastendruck- oder Bewegungsmelder-getriggert). `
      Da hat pix sehr recht. Alles andere wird schnell unübersichtlich. Also habe ich mich als Laie mal ans große Programmieren gemacht. Und siehe da, es klappt. Natürlich habe ich sehr viel auf die Skripte von pix und Sven als Basis zurückgegriffen.

      Hier also nun das Ergebnis der "Zerlegung" in Einzelskripts.

      1. zunächst stelle ich mit diesem Teilskript die "Basisinformationen" zusammen.

      ! // ##################################################################
      ! // Basisansage
      ! // ##################################################################
      ! // ##################################################################
      ! // Definitionen
      ! // ##################################################################
      ! function Aktualisieren () {
      ! // Quellen
      ! var idWetter = "hm-rega.0.29954";
      ! var idTemperatursensor = "hm-rpc.0.LEQ0177463.1.TEMPERATURE"; /Temperatursensor:1.TEMPERATURE/
      ! // Datenpunkt festlegen
      ! createState('Ansage.Basis', '', {
      ! name: 'Basisinformationen',
      ! type: 'string'
      ! });
      ! var tempAnsage = 'Ansage.Basis';
      ! function ermitteleAnsagedatum () {
      ! //Wochentag ermitteln
      ! var d = new Date ();
      ! var w = new Array("Sonntag","Montag","Dienstag","Mittwoch","Donnerstag","Freitag","Samstag");
      ! var wochentag = w[d.getDay()];
      ! //Tagesdatum ermitteln
      ! var tag = d.getDate();
      ! //Monat ermitteln
      ! var month = new Array("Januar","Februar","März","April","Mai","Juni","Juli","August","September","Oktober","November","Dezember");
      ! var monat = month[d.getMonth()];
      ! //Jahr ermitteln
      ! var jahr = d.getFullYear();
      ! //Stunde ermitteln
      ! var stunde = d.getHours();
      ! //Minute ermitteln
      ! var minute = d.getMinutes();
      ! return { // zurückgeben
      ! 'Jahr' : jahr,
      ! 'Monat' : monat,
      ! 'Tag' : tag,
      ! 'Wochentag' : wochentag,
      ! 'Stunde' : stunde,
      ! 'Minute' : minute
      ! };
      ! }
      ! function ermitteleWetter () { // Wetterbedingungen
      ! var wetterdaten = getState(idWetter).val;
      ! // log('Wetter: ' + wetterdaten);
      ! return(wetterdaten);
      ! }
      ! function ermitteleAnsageTemperatur () {
      ! // Die Außentemperatur ist xx. Das Wetter heute ist xx
      ! // Einfache Temperaturansage mit SayIt.
      ! // Variante 1 mittels splitten der Temperatur, damit die Ansage nicht
      ! // "Es sind 18 Punkt 2 Grad " sagt.
      ! var temperatursensor = getState(idTemperatursensor).val;
      ! var temp_string = temperatursensor.toString();
      ! // log('Temp ' + temp_string);
      ! var temp_array = [];
      ! temp_array = temp_string.split(".");
      ! // Fange leere Nachkommastellen ab. Das passiert, wenn die Temperatur z. B. 18.0 ist.
      ! // Es wird dann nur "18" gelesen.
      ! if (!temp_array[1]) {
      ! temp_array[1] = "0";
      ! // log("Die Nach-Kommastelle in temp_array[1] war nicht vorhanden und wird nun fest auf 0 gesetzt.");
      ! }
      ! return{
      ! 'Ganzzahl' : temp_array[0],
      ! 'Nachkommazahl' : temp_array[1]
      ! };
      ! }
      ! // ##################################################################
      ! // Basisansage
      ! // ##################################################################
      ! var ansagetext = "Guten Morgen, heute ist " + ermitteleAnsagedatum().Wochentag + " der " + ermitteleAnsagedatum().Tag + "te " + ermitteleAnsagedatum().Monat + ' ' + ermitteleAnsagedatum().Jahr
      ! + ". Es ist " + ermitteleAnsagedatum().Stunde + " Uhr und " + ermitteleAnsagedatum().Minute + " Minuten."
      ! + " Die Aussentemperatur beträgt " + ermitteleAnsageTemperatur().Ganzzahl + "," + ermitteleAnsageTemperatur().Nachkommazahl + " Grad."
      ! + " Wetter Bedingungen, " + ermitteleWetter() +" .";
      ! setState(tempAnsage, ansagetext);
      ! }
      ! // ##################################################################
      ! // Daten aktualisieren
      ! // ##################################################################
      ! // bei Ansageakualisier
      ! on ({id:"javascript.0.Ansage.Aktualisierer"/Basisdaten aktualisieren/, change: 'any'}, function(data) {
      ! Aktualisieren();
      ! });
      ! // und zurückstellen
      ! var tempAktualisierer = 'Ansage.Aktualisierer';
      ! setState(tempAktualisierer, "");

      Als nächstes werden die heutige Geburtstage mit folgendem Skript geholt:

      ! /* #######################################################
      ! Kalenderevent auswerten –--> nur Geburtstage
      ! sucht im iCal Adapter nach events (heute)
      ! ######################################################*/
      ! // Datenpunkt festlegen
      ! createState('Ansage.GeburtstageHEUTE', '', {
      ! name: 'Geburtstage ansagen',
      ! type: 'string'
      ! });
      ! var tempAnsage = 'Ansage.GeburtstageHEUTE';
      ! //Datum als String ermitteln;
      ! function ermitteleDatum() {
      ! var d= new Date();
      ! //Tagesdatum ermitteln
      ! var day = new Array("00","01","02","03","04","05","06","07","06","09","10","11","12","13","14","15","16","17","18","19","20","21","22","23","24","25","26","27","28","29","30","31");
      ! var t = day[d.getDate().toString()];
      ! //Monat ermitteln
      ! var month = new Array("01","02","03","04","05","06","07","08","09","10","11","12");
      ! var m = month[d.getMonth().toString()];
      ! //Jahr ermitteln
      ! var j = d.getFullYear().toString();
      ! var datum= t+"."+m+"."+j+" ";
      ! //log("datum: " + datum);
      ! return (datum);
      ! }
      ! function pruefeKalender() {
      ! var inhalt = getState('ical.0.data.table').val;
      ! var heute = ermitteleDatum();
      ! //log("HEUTE:" + heute) ;
      ! try{
      ! var ereignisse = JSON.stringify(inhalt, null, 2); // Ausgabe als String
      ! //log(ereignisse);
      ! var ereignisheute = '', // Liste (kommasepariert)
      ! ereignisheute_zeilen = ''; // Liste (mit Zeilenumbruch, zB für Anzeige in VIS)
      ! for(var i = 0; i <inhalt.length; i++)/{/alle/events/durchgehen<br="">if ( (inhalt__.date.indexOf(heute) != -1) || (inhalt__.date.indexOf('Heute') != -1) ) { // Strings Datum oder relatives Datum (nicht nicht) gefunden
      ! var ereignis = inhalt__.event;
      ! ereignis = ereignis.replace(',',''); // Komma im Namen ersetzen
      ! var komma = (i>0) ? ', ' : '';
      ! ereignisheute = ereignisheute + komma + ereignis;
      ! ereignisheute = ereignisheute.replace('Geburtstag von ', ''); // "Geburtstag von " löschen
      ! }
      ! }
      ! // Aufbereitung für die Ansage (falls vorhanden, wird letztes Komma durch und ersetzt)
      ! var lastkomma = ereignisheute.lastIndexOf(','); // letztes Komma in der Reihe
      ! if (lastkomma != -1) {
      ! var vorn = ereignisheute.slice(0,lastkomma);// lastkomma geändert in Null
      ! var hinten = ereignisheute.slice(lastkomma+1, ereignisheute.length);
      ! ereignisheute = vorn + ' und' + hinten;
      ! }
      ! // Ende Aufbereitung für die Ansage
      ! setState(tempAnsage, ereignisheute);
      ! //log('Geburtstage: ' + ereignisheute);
      ! } catch (fehler_try) {
      ! log('Fehler beim Kalenderevent einlesen ' + fehler_try);
      ! }
      ! }
      ! // bei Aktualisierung des Kalenders
      ! on ({id:'ical.0.events.Geburtstag', change: 'any'}, function(data) {
      ! pruefeKalender();
      ! });
      ! //bei Skriptstart
      ! schedule("40 2 * * *", function () {
      ! //log("===>Wird einmal am Tag ausgelöst");
      ! pruefeKalender();
      ! });______</inhalt.length;> ___Dann die heutigen Müllereignisse mit diesem Skript:

      ! // #################################################################
      ! // Müllkalender (HEUTE)
      ! // #################################################################
      ! createState('Ansage.MuellHEUTE', '', {
      ! name: 'Müll-Events heute',
      ! type: 'string'
      ! });
      ! var tempAnsage = 'Ansage.MuellHEUTE';
      ! //Datum als String ermitteln;
      ! function ermitteleDatum() {
      ! var d= new Date();
      ! //Tagesdatum ermitteln
      ! var day = new Array("00","01","02","03","04","05","06","07","06","09","10","11","12","13","14","15","16","17","18","19","20","21","22","23","24","25","26","27","28","29","30","31");
      ! var t = day[d.getDate().toString()];
      ! //Monat ermitteln
      ! var month = new Array("01","02","03","04","05","06","07","08","09","10","11","12");
      ! var m = month[d.getMonth().toString()];
      ! //Jahr ermitteln
      ! var j = d.getFullYear().toString();
      ! var datum= t+"."+m+"."+j+" ";
      ! //log("datum: " + datum);
      ! return (datum);
      ! }
      ! function pruefeKalender() {
      ! var inhalt = getState('ical.1.data.table').val;
      ! var heute = ermitteleDatum();
      ! try{
      ! var ereignisse = JSON.stringify(inhalt, null, 2); // Ausgabe als String
      ! log(ereignisse);
      ! var ereignisheute = '', // Liste (kommasepariert)
      ! ereignisheute_zeilen = ''; // Liste (mit Zeilenumbruch, zB für Anzeige in VIS)
      ! for(var i = 0; i <inhalt.length; i++)/{/alle/events/durchgehen<br="">if ( (inhalt
      .date.indexOf(heute) != -1) || (inhalt
      _.date.indexOf('Heute') != -1) ) { // Strings Datum oder relatives Datum (nicht nicht) gefunden
      ! var ereignis = inhalt__.event;
      ! ereignis = ereignis.replace(',',''); // Komma im Namen ersetzen
      ! var komma = (i>0) ? ', ' : '';
      ! ereignisheute = ereignisheute + komma + ereignis;
      ! }
      ! }
      ! // Aufbereitung für die Ansage (falls vorhanden, wird letztes Komma durch und ersetzt)
      ! var lastkomma = ereignisheute.lastIndexOf(','); // letztes Komma in der Reihe
      ! if (lastkomma != -1) {
      ! var vorn = ereignisheute.slice(0,lastkomma);// lastkomma geändert in Null
      ! var hinten = ereignisheute.slice(lastkomma+1, ereignisheute.length);
      ! ereignisheute = vorn + ' und' + hinten;
      ! }
      ! // Ende Aufbereitung für die Ansage
      ! setState(tempAnsage, ereignisheute);
      ! log('Müll: ' + ereignisheute);
      ! } catch (fehler_try) {
      ! log('Fehler beim Müll-Kalenderevent einlesen ' + fehler_try);
      ! }
      ! }
      ! // bei Aktualisierung des Kalenders
      ! on ({id:'ical.1.events.Abfuhr', change: 'any'}, function(data) {
      ! pruefeKalender();
      ! });
      ! //bei Skriptstart
      ! schedule("45 2 * * *", function () {
      ! log("===>Wird einmal am Tag ausgelöst");
      ! pruefeKalender();
      ! });___</inhalt.length;> ___Schließlich die Müllereignisse MORGEN hiermit (im Gegensatz zu den vorher veröffentlichten Skripten gibt es hier keinen Suchstring - es wird das angesagt, was im Kalender steht):

      ! // #################################################################
      ! // Müllkalender (ZUKUNFT)
      ! // #################################################################
      ! createState('Ansage.MuellZUKUNFT', '', {
      ! name: 'Müll-Events ZUKUNFT',
      ! type: 'string'
      ! });
      ! var tempAnsage = 'Ansage.MuellZUKUNFT';
      ! // User Anpassungen
      ! var stichtag = 1; // 0 heute; 1 morgen; 2 übermorgen; 3 In 3 Tagen
      ! var idTabelle = 'ical.2.data.table'; // Instanz eintragen
      ! // Ende User Anpassungen
      ! function ermitteleDatum() {
      ! var jetzt = new Date();
      ! log('JETZTZEIT: ' + jetzt);
      ! var zeit = new Date (jetzt.getFullYear(), jetzt.getMonth(), jetzt.getDate() + stichtag); // in drei Tagen um Mitternacht
      ! log('ZIELZEIT: ' + zeit);
      ! var jahr = zeit.getFullYear();
      ! var monat = (zeit.getMonth()+1 < 10) ? '0' + (zeit.getMonth()+1) : zeit.getMonth()+1;
      ! var tag = (zeit.getDate() < 10) ? '0' + zeit.getDate() : zeit.getDate();
      ! return (tag + '.' + monat + '.' + jahr);
      ! }
      ! function pruefeKalender() {
      ! var inhalt = getState('ical.2.data.table').val;
      ! var heute = ermitteleDatum();
      ! try{
      ! var ereignisse = JSON.stringify(inhalt, null, 2); // Ausgabe als String
      ! var ereignisheute = '';// Liste (kommasepariert)
      ! for(var i = 0; i <inhalt.length; i++)/{/alle/events/durchgehen<br="">if ( (inhalt
      .date.indexOf(heute) != -1) || (inhalt
      .date.indexOf('Heute') != -1) ) { // Strings Datum oder relatives Datum (nicht nicht) gefunden
      ! var ereignis = inhalt
      .event;
      ! ereignis = ereignis.replace(',',''); // Komma im Namen ersetzen
      ! var komma = (i>0) ? ', ' : '';
      ! ereignisheute = ereignisheute + komma + ereignis;
      ! }
      ! }
      ! // Aufbereitung für die Ansage (falls vorhanden, wird letztes Komma durch und ersetzt)
      ! var lastkomma = ereignisheute.lastIndexOf(','); // letztes Komma in der Reihe
      ! if (lastkomma != -1) {
      ! var vorn = ereignisheute.slice(0,lastkomma);// geändert, jetzt ohne +1 versus heute
      ! var hinten = ereignisheute.slice(lastkomma+1, ereignisheute.length);
      ! ereignisheute = vorn + ' und' + hinten;
      ! }
      ! // Ende Aufbereitung für die Ansage
      ! setState(tempAnsage, ereignisheute);
      ! log('Müll morgen: ' + ereignisheute);
      ! } catch (fehler_try) {
      ! log('Fehler beim Müll-Kalenderevent einlesen ' + fehler_try);
      ! }
      ! }
      ! // bei Aktualisierung des Kalenders
      ! on ({id:'ical.2.events.AbfuhrMORGEN', change: 'any'}, function(data) {
      ! pruefeKalender();
      ! });
      ! //bei Skriptstart
      ! schedule("50 2 * * *", function () {
      ! log("===>Wird einmal am Tag ausgelöst");
      ! pruefeKalender();
      ! });</inhalt.length;> ___Und hier das Skript zur Ansage über Sonos:

      ! // ##################################################################
      ! //
      ! // Ansagetext zusammenstellen
      ! //
      ! // ##################################################################
      ! // Definition Sayit für "2" = Buero;
      ! var idSayIt = "sayit.2.tts.text";
      ! var lautstaerke = 40;
      ! // Ansagetext
      ! // Basisansage
      ! var tempAnsage = "javascript.0.Ansage.Basis";
      ! var Ansage = getState(tempAnsage).val;
      ! //Geburtstagsansage
      ! var tempGEBURTSTAGE = "javascript.0.Ansage.GeburtstageHEUTE";
      ! var AnsageGEBURTSTAGE = getState(tempGEBURTSTAGE).val;
      ! //Muellsansage heute
      ! var tempMUELLheute = "javascript.0.Ansage.MuellHEUTE";
      ! var AnsageMUELLheute = getState(tempMUELLheute).val;
      ! //Muellsansage Zukunft
      ! var tempMUELLzukunft = "javascript.0.Ansage.MuellZUKUNFT";
      ! var AnsageMUELLzukunft = getState(tempMUELLzukunft).val;
      ! /log("Tempansage–------------------->" + tempAnsage);
      ! log("Ansage--------------------->" + Ansage);
      ! log("tempGEBURTSTAGE--------------------->" + tempGEBURTSTAGE);
      ! log("AnsageGEBURTSTAGE--------------------->" + AnsageGEBURTSTAGE);
      ! log("tempMUELLheute--------------------->" + tempMUELLheute);
      ! log("AnsageMUELLheute--------------------->" + AnsageMUELLheute);
      ! log("tempMUELLzukunft--------------------->" + tempMUELLzukunft);
      ! log("AnsageMUELLzukunft--------------------->" + AnsageMUELLzukunft);
      /
      ! // ##################################################################
      ! //
      ! // Basisdaten aktualisieren
      ! //
      ! // ##################################################################
      ! createState('Ansage.Aktualisierer', '', {
      ! name: 'Basisdaten aktualisieren',
      ! type: 'string'
      ! });
      ! var tempAktualisierer = 'Ansage.Aktualisierer';
      ! setState(tempAktualisierer, "1");
      ! // ####################################################
      ! // Wenn Geburtstag dann mit ansagen
      ! // ####################################################
      ! if (AnsageGEBURTSTAGE.length > 2) { // wenn der Inhalt des Objektes "AnsageGEBURTSTAGE" größer als 2 Zeichen lang ist, dann ...
      ! Ansage = Ansage + "Heute haben Geburtstag, " + AnsageGEBURTSTAGE +" .";
      ! //log("Ansage mit Geburtstag--------------------->" + Ansage);
      ! }
      ! // ####################################################
      ! // Wenn Müll dann mit ansagen
      ! // ####################################################
      ! if (AnsageMUELLheute.length > 2) { // wenn der Inhalt des Objektes "AnsageMUELLheute" größer als 2 Zeichen lang ist, dann ...
      ! Ansage = Ansage + " Achtung, heute ist Abholung " + AnsageMUELLheute +" ." + "Bitte an die Strasse stellen" + " !";
      ! //log("Ansage mit Geburtstag und Muell heute--------------------->" + Ansage);
      ! }
      ! // ####################################################
      ! // Wenn Müll ZUKUNFT, dann mit ansagen
      ! // ####################################################
      ! if (AnsageMUELLzukunft.length > 2) { // wenn der Inhalt des Objektes "AnsageMUELLzukunft" größer als 2 Zeichen lang ist, dann ...
      ! Ansage = Ansage + " Achtung, morgen ist Abholung " + AnsageMUELLzukunft + " ." + "Bitte MORGEN an die Strasse stellen" + " !";
      ! //log("Ansage mit Geburtstag und Muell heute und morgen--------------------->" + Ansage);
      ! }
      ! // ####################################################
      ! // ANSAGE
      ! // ####################################################
      ! setState(idSayIt, lautstaerke + ";" + Ansage);
      _________Noch ein paar kleine Hinweise:

      1. Die Müll- und Geburtstagskripte werden mit der "schedule" Anweisung einmal nachts und mittels "on" Anweisung bei Veränderung aktualisiert.

      2. Die "Basisdaten" werden aktualisiert, wenn das "Ansageskript" ausgelöst wird. Dazu habe ich ein Objekt "Aktualisierer" angelegt, auf welches das "Basis" Skript reagiert und vom Ansageskript ausgelöst wird.

      3. Zumindest in den mir vorliegenden und aus dem Forum "entwendeten" Skripten war in dem Teil "// Aufbereitung für die Ansage (falls vorhanden, wird letztes Komma durch und ersetzt)" noch ein Fehler. An der Stelle_________

      var vorn = ereignisheute.slice(0,lastkomma);// geändert, jetzt ohne +1
      

      stand hinter "lastkomma" ein "+1", das bei mir zu komischen Ergebnissen geführt hat.

      So, und nun bin ich auf Eure Hinweise und Verbesserungsvorschläge mehr als gespannt.

      Lieber Skorpil,

      du schreibst, dass der Müll Zukunft ohne besonderen String gesucht wird (so verstehe ich das zumindest)? Ich bin zu blöd, den Unterschied in den Codes zu finden. Habe sie mir nebeneinander gelegt, wo genau ist der entscheidende Unterschied, dass er bei Müll Zukunft auswertet, was im Kalender steht, während sonst das Event getagged sein muss? Oder verstehe ich dich da völlig falsch?

      Generell: gibt es etwas zum Thema, dass bei ical.data.table immer nur [object] [object] ausgegeben wird? Das scheint bei mir das Hauptproblem zu sein. der html Teil ist korrekt befüllt, beim anderen steht immer nur das vorgenannte object Thema und es kommt nichts…

      Danke und Grüße vom Anfänger

      tempestas_________ `

      1 Reply Last reply Reply Quote 0
      • S
        skorpil last edited by

        Tempestas,

        Welche Codes hast Du denn verglichen? Ich wiederum verstehe Deine Frage nämlich nicht genau (ist ja auch schon wieder eine Weile her, dass ich das gepostet habe. Muss mich erst wieder reindenken).

        In machen Posts vorher wurde immer nach einem expliziten Ereignis gesucht. Daher musste ein expliziter Suchstring auch definiert werden. Ich habe dagegen einen Kalender angelegt, der NUR Müllereignisse enthält. Und dann schaut er nur, was morgen drinsteht - soweit ich mich erinnere. Wenn ein Skript dann läuft, vergisst man schnell die Details.

        Das mit object und html war mir auch aufgefallen. Da meine Skripts aber auf denen von Pix basieren und Pix das mit object programmiert hat, habe ich daran nicht gerührt.

        Liebe Grüße

        Bernd

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

          Hallo Bernd,

          ich hatte mich bezogen auf die von mir zitieren Skripte deinerseits.

          Ich hatte gedacht, dass im "Müll heute" Skript eine Art definierter Suchstring sein muss, den es im Skript "Müll Zukunft" nicht gibt. Diesen Unterschied finde ich allerdings nicht. Oder meintest du, dass du in der jeweiligen iCal Instanz das Event nicht angelegt hast für die zukünftigen Mülltermine?

          Letztlich geht es mir ja auch darum, dass ich ohne ein "event" zu benennen aus verschiedenen Kalendern, die nur dezidiert für diese Aufgabe geschaffen sind, die täglichen Eintragungen auszulesen.

          Ich möchte aber eben nicht jeden Termin "Termin XYZ" nennen müssen und im iCal das event "Termin" definieren sondern einfach nur "XYZ" eintragen und dann soll er mir morgens im Bad sagen "heute steht XYZ an" (für jeden Termin, nicht nur ganztägige).

          Gleiches mit den Geburtstagen. Mir reicht "Klaus Müller" statt "Geburtstag Klaus Müller" im Kalender.

          Aus mir unerfindlichen Gründen wird bei mir aber nichts ausgewertet aus den iCals; ob das an dem Thema ical.0.data.table = [object] [object] liegt weiß ich nicht, wäre aber meine Vermutung.

          Danke und Grüße

          Steffen

          1 Reply Last reply Reply Quote 0
          • S
            skorpil last edited by

            @tempestas:

            Hallo Bernd,

            ich hatte mich bezogen auf die von mir zitieren Skripte deinerseits.

            Ich hatte gedacht, dass im "Müll heute" Skript eine Art definierter Suchstring sein muss, den es im Skript "Müll Zukunft" nicht gibt. Diesen Unterschied finde ich allerdings nicht. Oder meintest du, dass du in der jeweiligen iCal Instanz das Event nicht angelegt hast für die zukünftigen Mülltermine?

            Letztlich geht es mir ja auch darum, dass ich ohne ein "event" zu benennen aus verschiedenen Kalendern, die nur dezidiert für diese Aufgabe geschaffen sind, die täglichen Eintragungen auszulesen.

            Ich möchte aber eben nicht jeden Termin "Termin XYZ" nennen müssen und im iCal das event "Termin" definieren sondern einfach nur "XYZ" eintragen und dann soll er mir morgens im Bad sagen "heute steht XYZ an" (für jeden Termin, nicht nur ganztägige).

            Gleiches mit den Geburtstagen. Mir reicht "Klaus Müller" statt "Geburtstag Klaus Müller" im Kalender.

            Aus mir unerfindlichen Gründen wird bei mir aber nichts ausgewertet aus den iCals; ob das an dem Thema ical.0.data.table = [object] [object] liegt weiß ich nicht, wäre aber meine Vermutung.

            Danke und Grüße

            Steffen `
            Lieber Steffen,

            Du hast schon recht: in den beiden Müllskripts gibt es keinen besonderen Suchstring. Es wird das angesagt, was drin steht. Warum das bei Dir nicht klappt, weiß ich nicht. Außer vielleicht: hast Du bei der Konfiguration der iCal Instanz auch einen Namen für den Kalender vergeben? Bei mir heißt der Kalender Abfuhr. Bin nicht mehr ganz sicher, ob das eine Rolle spielt bei der Suche…

            Bin jetzt erst einmal offline. Bei Bedarf melde Dich gerne heute Abend noch mal.

            Liebe Grüße

            Bernd

            1 Reply Last reply Reply Quote 0
            • N
              Nanouk last edited by

              Hallo Tempestas,

              ursprünglich war es mal so:

              `/* Kalenderevent in der Zukunft suchen
              
              sucht im iCal Adapter nach events, die noch kommen
              http://forum.iobroker.net/viewtopic.php?f=21&t=3761&sid=4319378b32a0ce096bbbbfc0ebd859ce#p34975
              
              20.09.2016 erstellt von pix
              13.11.2016 angepasst für Sven
              */
              
              // User Anpassungen
              var suchstring =   'restmüll';              // Suchbegriff
              var stichtag =     1;                   // 0 heute;  1 morgen; 2 übermorgen; 3 In 3 Tagen
              var idTabelle =    'ical.5.data.table'; // Instanz eintragen
              var logging =      true;                // debug Log ein/ausschalten
              // Ende User Anpassungen
              
              createState('Muellereignisse.' + suchstring, false, {
                   type: 'boolean',
                   name: suchstring + ' wird morgen abgeholt',
                   desc: suchstring + ' wird morgen abgeholt (Boolean)',
                   def: false
              });
              var idEventState = getState('Muellereignisse.' + suchstring).val; // am 27.11.2016 editiert
              
              var tage = ['Heute','Morgen','Übermorgen','In 3 Tagen']; // dafür relative Datumsangabe in iCal-Adapter einschalten und ggf. übersetzen
              
              function datum(x) {
                  var jetzt = new Date();
                  if (logging) log('Jetztzeit: ' + jetzt);
                  //var zeit = new Date(jetzt.getTime() + x * 24 * 60 * 60 * 1000); // es werden genau x Tage, als x * 24 Stunden zugezählt --> nicht gut
                  var zeit = new Date (jetzt.getFullYear(), jetzt.getMonth(), jetzt.getDate() + x); // in drei Tagen um Mitternacht
                  if (logging) log('Zielzeit: ' + zeit);
                  var jahr       = zeit.getFullYear();
                  var monat      = (zeit.getMonth()+1 < 10) ? '0' + (zeit.getMonth()+1) : zeit.getMonth()+1;
                  var tag        = (zeit.getDate() < 10) ? '0' + zeit.getDate() : zeit.getDate();
                  return (tag + '.' + monat + '.' + jahr);
              }
              
              function pruefeKalender() {
                  var kalender = getState(idTabelle).val;
                  var tag = datum(parseInt(stichtag,10));
                  if (logging) log(tag);
                  try{
                      var ereignisse = JSON.stringify(kalender, null, 2); // Ausgabe als String
                      if (logging) log(ereignisse);
                      for(var i = 0; i <kalender.length; 18/i++)/{/alle/events/durchgehen/if/(/(kalender[i].date.indexof(tag)/!="-1)" ||/(kalender[i].date.indexof(tage[stichtag])/)/string/datum/oder/relatives/(nicht/nicht)/gefunden,/also/gefunden/var/termin="kalender[i].event;" (termin.indexof(suchstring)/log('treffer:/'/+/tage[stichtag]/suchstring);/hier/wird/später/vielleicht/stichtag_array/verwendet/setstate(ideventstate,/true);/}/else/false);/catch/(fehler_try)/log('fehler/beim/kalenderevents/einlesen/fehler_try,/'error');/schedule("30/*/*",/function(){/bei/mehreren/müllskripts/den/ersten/wert/verändern/(minuten)/event="getState(idEventState).val;" (event)/ansage="suchstring" morgen/abholt./bitte/rausstellen!';/kann/sayit/push/gesetzt/werden/setstate("sayit.0.tts.text"/*text/to/speech*/,/ansage);/});/aktualisierung/on/({id:/idtabelle,/change:/'any'},/function(data)/pruefekalender();/1min/nach/mitternacht/schedule("1/pruefekalender);/skriptstart/pruefekalender();<e=""></kalender.length;>`
              
              Da Skorpil aber einen eigenen Kalender für z.B. Müll pflegt, hat er auf den Suchstring verzichtet und lässt sich alles aus dem Kalender ausgeben.
              
              Daher findest Du in seinem angepassten Skript den Suchstring nicht.
              
              Korregiert mich wenn ich falsch liege.
              
              Gruss Nanouk[/i][/i][/i]
              
              1 Reply Last reply Reply Quote 0
              • T
                tempestas last edited by

                Vielen Dank euch beiden.

                Das mit dem Suchstring habe ich zwischenzeitlich auch gefunden gehabt. Ich versuche immer, alle irgendwie artverwandten Threads zu sichern, aber manchmal entdecke ich sie dann doch wieder "neu" 😉

                Da mit dem Kalendernamen prüfe ich nochmal. Also nen Namen haben beide, aber soweit ich das sehe wird doch immer die direkte Instanz angesprochen (ical.0.data.table z.B.) und nicht der Name des Kalenders?

                Werde ich heute abend prüfen, ob mich das weiterbringt.

                VG an beide und danke

                Nachtrag:

                könnte mir jemand erklären, was hier genau passiert?

                createState('Ansage.MuellZUKUNFT', '', {

                name: 'Müll-Events ZUKUNFT',

                type: 'string'

                });

                createState erzeugt ein Object, soweit klar. (bilde ich mir ein 🙂

                Was genau wird mit dem "Ansage.MuellZUKUFNT" gemacht?

                Warum wird der name definiert, ich sehe nirgends, dass das relevant ist (oder darf es nur einfach nicht leer bleiben?)?.

                Auf meinen üblichen Nachschlageseiten (w3schools) finde ich leider nichts dazu und das Github sagt zwar etwas, aber dort ist die Schreibweise irgendwie anders. Ich bekomme das nicht zusammen.

                1 Reply Last reply Reply Quote 0
                • N
                  Nanouk last edited by

                  Der Name wird im Admin unter den Objekten angezeigt. Also nach der ID.

                  createState erzeugt das Objekt und das erzeugte Objekt wird dann mit dem Kalendereintrag befüllt.

                  Da ist bei mir noch der Knackpunkt.

                  Objekt MuellHEUTE wird richtig beschrieben, MuellZUKUNFT jedoch nur mit "heutigem Kalendereintrag".

                  Ich such mir nen Wolf und finde den Fehler nicht.

                  :mrgreen:

                  Nachtrag:

                  Ich glaube weiter ist der Name nicht relevant. Wenn der Teil nicht vorhanden ist, wird der Name im Admin unter Objekten genauso angezeigt wie die ID. Also MuellZUKUNFT.

                  Ich poste mal mein aktuelles Skript, vielleicht kann mir jemand sagen wo der Fehler liegt, dass statt dem morgigen Kalendereintrag immer der heutige ins Objekt geschrieben wird:

                  `// #################################################################
                  
                  // Müllkalender (ZUKUNFT)
                  
                  // #################################################################
                  
                  createState('Ansage.MuellZUKUNFT', '', {
                  name: 'Müll-Events ZUKUNFT',
                  type: 'string'
                  });
                  
                  var tempAnsage = 'Ansage.MuellZUKUNFT';
                  
                  // User Anpassungen
                  var stichtag = 1; // 0 heute; 1 morgen; 2 übermorgen; 3 In 3 Tagen
                  var idTabelle = 'ical.5.data.table'; // Instanz eintragen
                  // Ende User Anpassungen
                  
                  function ermitteleDatum() {
                  var jetzt = new Date();
                  log('JETZTZEIT: ' + jetzt);
                  
                  var zeit = new Date (jetzt.getFullYear(), jetzt.getMonth(), jetzt.getDate() + stichtag); // in drei Tagen um Mitternacht
                  
                  log('ZIELZEIT: ' + zeit);
                  var jahr = zeit.getFullYear();
                  var monat = (zeit.getMonth()+1 < 10) ? '0' + (zeit.getMonth()+1) : zeit.getMonth()+1;
                  var tag = (zeit.getDate() < 10) ? '0' + zeit.getDate() : zeit.getDate();
                  return (tag + '.' + monat + '.' + jahr);
                  }
                  
                  function pruefeKalender() {
                  var inhalt = getState('ical.5.data.table').val;
                  var heute = ermitteleDatum();
                  try{
                  var ereignisse = JSON.stringify(inhalt, null, 2); // Ausgabe als String
                  var ereignisheute = '';// Liste (kommasepariert)
                  
                  for(var i = 0; i <inhalt.length; i++)/{/alle/events/durchgehen/if/(/(inhalt[i].date.indexof(heute)/!="-1" )/||/(inhalt[i].date.indexof('heute')/strings/datum/oder/relatives/(nicht/nicht)/gefunden/var/ereignis="inhalt[i].event;" komma/im/namen/ersetzen="">0) ? ', ' : ''; 
                  ereignisheute = ereignisheute + komma + ereignis;
                  } 
                  }
                  // Aufbereitung für die Ansage (falls vorhanden, wird letztes Komma durch und ersetzt)
                  var lastkomma = ereignisheute.lastIndexOf(','); // letztes Komma in der Reihe
                  if (lastkomma != -1) {
                  var vorn = ereignisheute.slice(0,lastkomma);// geändert, jetzt ohne +1 versus heute
                  var hinten = ereignisheute.slice(lastkomma+1, ereignisheute.length);
                  ereignisheute = vorn + ' und' + hinten;
                  }
                  // Ende Aufbereitung für die Ansage
                  
                  setState(tempAnsage, ereignisheute);
                  log('Müll morgen: ' + ereignisheute);
                  } catch (fehler_try) {
                  log('Fehler beim Müll-Kalenderevent einlesen ' + fehler_try);
                  }
                  }
                  
                  // bei Aktualisierung des Kalenders
                  on ({id:'ical.5.events.AbfuhrMORGEN', change: 'any'}, function(data) {
                  pruefeKalender();
                  });
                  
                  //bei Skriptstart
                  schedule("*/1 * * * *", function () {
                  log("===>Wird einmal am Tag ausgelöst"); 
                  pruefeKalender();
                  });</inhalt.length;>`
                  
                  Schedule sitzt so kurz weil ich am ausprobieren bin.[/i][/i][/i]
                  
                  1 Reply Last reply Reply Quote 0
                  • T
                    tempestas last edited by

                    ich versuche weiterhin, die nachfolgenden Teile zu verstehen und komme mit den Github Erläuterungen und den Tutorials die ich finde nicht weiter (bin BWLer, kein IT Mensch 🙂

                    Ich will es inhaltlich verstehen

                    createState('Ansage.MuellZUKUNFT', '', {
                    name: 'Müll-Events ZUKUNFT',
                    type: 'string'
                    });
                    
                    

                    Hier wird also das Objekt "Ansage.MuellZUKUFNT kreiirt, korrekt? (der Punkt ist unerheblich?)

                    Ein Name wird vergeben, der aber ansonsten wohl egal ist und der Typ auf String gesetzt. Soweit, so gut.

                    var tempAnsage = 'Ansage.MuellZUKUNFT';
                    

                    Hie erhält die Variable "tempAnsage" …was genau? Ich dachte, "Ansage.MuellZUKUNFT" ist lediglich die ID des zuvor erzeugten Objekts?

                    setState(tempAnsage, ereignisheute);
                    

                    hier wird der Status der tempAnsage mit dem Inhalt von ereignisheute beschrieben, korrekt? ereignisheute habe ich verstanden.

                    Wieso wird tempAnsage oben etwas zugeordnet, nur damit dann via setState etwas geändert wird? Oder interpretiere ich hier die Laufrichtung der Zuordnungen falsch?

                    Wozu überhaupt gibt es den Teil mit "Ansage.MuellZUKUNFT" ? Irgendwie sehe ich auch nicht, dass "Ansage.MuellZUKUNFT" irgendwo wieder angesprochen wird, es erhält einen Wert… und dann?

                    Gibt tempAnsage den inhalt von "ereignisheute" an das Object "Ansage.MuellZUKUNFT"? Wenn ja, habe ich wohl die Leserichtung missverstanden. Warum muss ich diesen Zwischenschritt gehen?

                    Wäre toll, wenn mir das jemand erklärt, gerne mit nem Beispielinhalt ("Klaus-Bärbel")

                    Dank und Gruß vom (totalen) Anfänger

                    1 Reply Last reply Reply Quote 0
                    • P
                      pix last edited by

                      Hallo tempestas,

                      es hat sich in den Anfangszeiten hier im Forum eine Schreibweise etabliert, die jetzt manchmal nicht eingehalten wird, da sie nicht offiziell und natürlich nicht vorgeschrieben ist.

                      Ich mache es so:

                      Der Datenpunkt eines Objektes (ein State) wird ja zB mit createState erstellt oder ist bei einem Aktor oder Sensor schon vorhanden. Wenn man ihn selbst erstellt, bewirken die Punkte die EInordnung in Ordner im Admin Reiter Objekte. Für die Übersichtlichkeit lassen sich so die Objekte, die für ein Skript gebraucht werden in einen Ordner packen.

                      Das hattest du ja schon verstanden.

                      createState('Pfad.Name', …

                      erstellt also ein Objekt mit dem Datenpunkt Name im Ordner Pfad (genau genommen ist es kein Pfad, sondern ein Teil des Namens, aber das ist hier erstmal egal). In diesem Datenpunkt soll eine Zeichenkette gespeichert werden. Dazu wird dieser Datenpunkt mit setState('Pfad.Name', 'Zeichenkette') oder besser setState('Adaptername.Instanznummer.Pfad.Name', 'Zeichenkette') beschrieben. Dieser Rattenschwanz kann ganz schön lang werden. Um das abzukürzen und den Code etwas übersichtlicher zu bekommen, kann man einfach den Namen, den Pfad, die Instanz und den Adapternamen als String in eine lokale Variable schreiben. In der Variable steht dann nur der Name (zB 'javascript.0.Flur.Licht'). Und hier kommt die oben erwähnte Schreibweise ins Spiel: Solche Variablen beginnen nach dieser Schreibweise mit 'id'. Also zB idFlurlicht = 'javascript.0.Flur.Licht'. Jetzt mal in Codetags:

                      createState('Flur.Licht', false, {
                         name: 'Flurlicht',
                         type: 'boolean',
                         def: false
                      });
                      
                      var idFlurlicht = 'javascript.0.Flur.Licht';
                      
                      setState(idFlurlicht, true);
                      // ist das gleiche wie 
                      setState('javascript.0.Flur.Licht', true);
                      
                      // und hier der richtige Wert
                      var flurlicht = getState(idFLurlicht).val;
                      
                      

                      Will man im Nachhinein mal die Ordnerstruktur ändern, muss man zB nicht den ganzen Code durchgehen ändern. Oder auch das Austauschen von Sensoren/Aktoren geht so leichter.

                      Ich versuche immer oben im Skript die Datenpunktnamen in Variablen mit id-Prefix zu schreiben. So kann ich auch beim Überfliegen des Codes schnell feststellen, welchen Variablen tatsächlich Werte enthalten (flurlicht, immer klein geschrieben) und welche nur einen Namen eines Datenpunktes (zB idFlurlicht, immer id vorweg und mit einem großen Buchstaben weiter)

                      Gruß

                      Pix

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

                        Hallo Pix,

                        vielen Dank, das hat mich schonmal weitergebracht.

                        Habe jetzt gelernt und begriffen, dass ich in den Objekten dann tatsächlich besagte Objekte finde 🙂

                        Aus mir unverständlichen Gründen funktioniert bei mir das Auslesen aus dem ical trotzdem nicht. Ich habe glaube ich inzwischen jedes Skript durch, dass hier jemals gepostet wurde. Immer das gleiche Ergebnis:

                        Die "Standardansage" also Tag, Datum, Uhrzeit und Temperatur/Wetter funktioniert.

                        Die Objekte, die für Müll (bei mir "Termine" aller Art) und Geburtstage (separater Kalender) erzeugt werden sind immer leer.

                        Als Zwischenschritt würde ich jetzt gerne erstmal versuchen, überhaupt etwas einzulesen, vor jeglicher Formattierung für Ansagen via Sayit. Zum Üben reicht mir ja auch der Geburtstagskalender.

                        Ausgangslage:

                        • ical.0 Instanz ist verlinkt auf meinen google Geburtstagskalender (manuell erzeugt mit Testdaten).

                        • in der Instanz selbst sehe ich auch bei html die Termine, bei table steht [object][object]

                        ich will keinen Suchstring nutzen, da ich das meiner Frau nicht werde dauerhaft beibringen können, dass sie immer "Geburtstag" oder "Termin" in die Kalender schreiben soll 😉

                        Mein Ziel, auch um das Vorgehen zu verstehen, wäre also ical.0.data.table auszulesen in eine Variable als String, korrekt? Dann müsste ich diese im Reiter Objekte ja entsprechend befüllt sehen können? Damit wäre mein erstes Zwischenziel erreicht.

                        Leider sitze ich gerade nicht am iobroker, kann also nur hier im Forum mal eintippen, was ich mir denke. Ich greife dabei auf das Skript von Skorpil/Pix zurück, dass Skorpil hier am 4.2.2017 gepostet hat Dazu schreibe ich als Kommentare mein Verständnis dessen, was da so passiert (oder auch nicht)

                        
                        /* #######################################################
                        Kalenderevent auswerten ----> nur Geburtstage  // Kalender hat nur Geburtstagsdaten, Suchstring entbehrlich
                        sucht im iCal Adapter nach events (heute)
                        ######################################################*/
                        
                        createState('Ansage.GeburtstageHEUTE', '', {
                        name: 'Geburtstage ansagen',
                        type: 'string'
                        });
                        
                        

                        Kommentar: Es wurde der der Datenpunkt erzeugt. Im Reiter "Objekte" kann ich unter dem Ordner "java.0" dann "Ansage" sehen und darin "GeburtstageHEUTE als Datenpunkt. Die Namenszuordnung ist für die Operation irrelevant, der Typ muss string sein zum Einlesen der Daten. Letztlich sollte ich im Objekte Reiter unter "State" später die Inhalte aus dem Kalender sehen können, oder?

                        var inhaltAnsage = 'Ansage.GeburtstageHEUTE';  
                        
                        

                        Mein Verständnis hier: inhaltAnsage wird später mit den Ereignissen aus dem Kalender befüllt. Wird hier dann dem Objekt Ansage.GeburtstageHEUTE, welches ja zuvor geschaffen wurde, der inhalt des Kalender zugeordnet / übermittelt? Meine "Leserichtung" würde mir eigentlich sagen, dass der Variablen inhaltAnsage hier das Objekt zugeordnet wird. Aber dieses ist doch aktuell leer? Hier brauche ich bitte nochmal eine Erklärung, das Einsteiger Tutorial war im Bereich Variablen dafür mich nicht ausreichend.

                        Jetzt fehlt also noch, dass der Kalender ausgelesen wird.

                        Hinweis: Den Teil mit "Datum als String ermitteln" aus Skorpils Skript habe ich verstanden und lasse ihn daher hier weg

                        function pruefeKalender() {
                        var inhalt = getState('ical.0.data.table').val;
                        var heute = ermitteleDatum();
                        
                        

                        In die Variable inhalt wird nun der Kalender ausgelesen

                        Variable heute enthält das zuvor als String ermittelte Datum

                        try{
                        var ereignisse = JSON.stringify(inhalt, null, 2); // Ausgabe als String
                        
                        

                        Variable ereignisse wird befüllt mit den ausgelesenen Werten aus dem Kalender als string (via inhalt)

                        var ereignisheute = '', // Liste (kommasepariert)
                        ereignisheute_zeilen = ''; // Liste (mit Zeilenumbruch, zB für Anzeige in VIS)
                        
                        

                        Hier wird formatiert. Warum ist "ereignisheute_zeilen" ohne var vorweg deklariert?

                        for(var i = 0; i <inhalt.length; i++)/{/alle/events/durchgehen/<e=""></inhalt.length;>
                        

                        Iteration wird eingeleitet

                        `if ( (inhalt[i].date.indexOf(heute) != -1) || (inhalt[i].date.indexOf('Heute') != -1) ) {` 
                        
                        wenn das Datum der i-ten Position von inhalt  .... [??]... des Strings heute ungleich -1  (=true) (verstehe ich nicht, i startet doch bei 0?), dann.... HELP :)
                        
                        `~~[code]~~var ereignis = inhalt[i].event;` 
                        
                        Variable ereignis erhält i-tes event.
                        
                        FRAGE: event stammt aus dem ical Adapter, korrekt? Muss ich also events doch anlegen (z.B. "Geburtstag")? oder meint event hier einen Eintrag generell? Wenn ja, muss im ical adapter dann events leer gelassen werden? 
                        
                        `~~[code]~~ereignis = ereignis.replace(',',''); // Komma im Namen ersetzen
                        var komma = (i>0) ? ', ' : '';
                        ereignisheute = ereignisheute + komma + ereignis;
                        ereignisheute = ereignisheute.replace('Geburtstag von ', ''); // "Geburtstag von " löschen
                        } 
                        }
                        [/code]`
                        
                        Formattierung. Frage: Das "Geburtstag von" wird gelöscht. D.h. für mich, dass Skorpil in seinem Kalender immer "Geburtstag von Klaus/Bärbel..." eingetragen hat. Wird hier also doch nach dem event "Geburtstag" gesucht? Oder geht es auch, wenn ich in meinem dezidierten Geburtstagskalender eben nur "Klaus" stehen habe? Ich würde dann die letzte Zeile entsprechend weglassen.
                        
                        `~~[code]~~// Aufbereitung für die Ansage (falls vorhanden, wird letztes Komma durch und ersetzt)
                        var lastkomma = ereignisheute.lastIndexOf(','); // letztes Komma in der Reihe
                        if (lastkomma != -1) {
                        var vorn = ereignisheute.slice(0,lastkomma);// lastkomma geändert in Null
                        var hinten = ereignisheute.slice(lastkomma+1, ereignisheute.length);
                        ereignisheute = vorn + ' und' + hinten;
                        }
                        [/code]`
                        
                        Formattierung und ereignisheute wird zusammengesetzt
                        
                        `~~[code]~~// Ende Aufbereitung für die Ansage
                        setState(inhaltAnsage, ereignisheute);
                        //log('Geburtstage: ' + ereignisheute); 
                        
                        } 
                        [/code]`
                        
                        hier wird nun endlich der inhaltAnsage der Status aus ereignisheute, welche die Daten aus dem Kalender enthält übergeben.
                        
                        `~~[code]~~//Geburtstagsansage 
                        
                        var idGeburtstage = "javascript.0.Ansage.GeburtstageHEUTE";  
                        var AnsageGeburtstage = getState(idGeburtstage).val;  
                        [/code]`
                        
                        Kommentar: zunächst folge ich mal Pix's Hinweis und bennenne die Variable in id um. Inhaltlich wird hier der Variablen der Pfad zugewiesen zur Verkürzung der nachfolgenden Programmierung. Es wird also der Variablen ansageGeburtstage dann via des verkürzten Pfades der Wert (State) aus "javascript.0.Ansage.GeburtstageHEUTE" zugeordnet, korrekt?
                        
                        Diesen haben wir oben mit inhaltAnsage beschrieben.
                        
                        Dies ist dann für die Sonos Ausgabe gedacht
                        
                        Habe ich das soweit i.W. richtig verstanden, wo liege ich falsch?
                        
                        Ich bedanke mich vorab bei allen, die mir auf die Sprünge helfen. Ich hoffe man merkt, dass ich mir Mühe gebe zu lernen und es selbständig zu durchdringen und nicht einfach stumpf das "gemachte Bett" haben will. 
                        
                        beste Grüße
                        
                        tempestas[/i][/code][/i][/i]
                        
                        1 Reply Last reply Reply Quote 0
                        • P
                          pix last edited by

                          @tempestas:

                          Hallo Pix,

                          vielen Dank, das hat mich schonmal weitergebracht. `
                          Das ist gut. 😛

                          @tempestas:

                          • in der Instanz selbst sehe ich auch bei html die Termine, bei table steht [object][object] `
                            Habe das gleiche Problem und deshalb das Skript bei mir vor einiger Zeit verworfen.

                          Pix

                          1 Reply Last reply Reply Quote 0
                          • S
                            skorpil last edited by

                            Pix,

                            Frage bzw. Hinweis:

                            1. das mit object object habe ich manchmal, aber nicht immer. Meist stehen Werte drin. Die sich dann ja gottlob auch weiter verarbeiten lassen. Denn sonst würde mein Script ja nicht funktionieren. Das tut es aber. Eine Logik, wann ja, wann nein habe ich noch nicht ausmachen können. Manchmal ist es nur eine Frage der Zeit. Dann steht was drin…

                            2. wenn wir also mit object immer wieder Probleme haben: warum lösen wir das nicht über html. Meine Programmierfähigkeiten reichen nicht aus, um das zu lösen. Tempestas hat das ja auch schon angeregt. Könntest Du Dich nicht erbarmen und das Script so umschreiben, dass wir einfach auf html zugreifen?

                            Liebe Grüße

                            Bernd

                            1 Reply Last reply Reply Quote 0
                            • R
                              RobS last edited by

                              Moin,

                              ich habe schon seit längerem das Skript mit Ansage von Wetter, Geburtstag und Müll als morgendlichen Wecker im Einsatz - vielen Dank dafür nochmal an Pix und Skorpil!

                              Seit einiger Zeit, bekomme ich, wenn jemand Geburtstag hat, die Ansage "heute hat hat Geburtstag Geburtstag" (also "hat Geburtstag" statt des Namens).

                              Ich habe das Skript nicht verändert und habe keine Ahnung welche sonstige Änderung diesen Fehler verursachen könnte.

                              Hier mal der Abschnitt im Skript um den es wohl geht, vielleicht findet dort jemand nen Fehler:

                              `/
                              * Bereich - Geburtstagskalender */
                              
                              createState('Kalenderereignisse.heute', '', {
                                  name: 'Events von heute',
                                  type: 'string'
                              });
                              
                              var idListe =  'Kalenderereignisse.heute';
                              
                              /* Kalenderevent auswerten ----> nur Geburtstage */
                              function pruefeGeburtstagskalender() {
                                  var inhalt = getState('ical.4.data.table').val;
                                  var inhaltanzahl = getState('ical.4.data.count').val;
                                  var heute = ermitteleDatum();
                                  try{
                                      var ereignisse = JSON.stringify(inhalt, null, 2); // Ausgabe als String
                                      log(ereignisse);
                                      var ereignisheute = '', // Liste (kommasepariert)
                                          ereignisheute_zeilen = ''; // Liste (mit Zeilenumbruch, zB für Anzeige in VIS)
                                      for(var i = 0; i <inhalt.length; i++)/{/alle/events/durchgehen/if/(/(inhalt[i].date.indexof(heute)/!="-1)" ||/(inhalt[i].date.indexof('heute')/)/strings/datum/oder/relatives/(nicht/nicht)/gefunden/var/ereignis="inhalt[i].event;" komma/im/namen/ersetzen="">0) ? ', ' : '';
                                              ereignisheute = ereignisheute + komma + ereignis;
                                              ereignisheute = ereignisheute.replace('hat Geburtstag', ''); // "hat Geburstag" löschen
                                          } 
                                      }
                                      // Aufbereitung für die Ansage (falls vorhanden, wird letztes Komma durch und ersetzt)
                                      var lastkomma = ereignisheute.lastIndexOf(', '); // letztes Komma in der Reihe
                                      if (lastkomma != -1) {
                                         var vorn = ereignisheute.slice(0,lastkomma-1);
                                         var hinten = ereignisheute.slice(lastkomma+1, ereignisheute.length);
                                         ereignisheute = vorn + ' und' + hinten + ' haben Geburtstag';
                                      }else{
                                          ereignisheute = ereignisheute + ' hat Geburtstag';
                                      }
                                      // Ende Aufbereitung für die Ansage
                              
                                      setState(idListe, ereignisheute);
                                      log('Geburtstage: ' + ereignisheute);
                                    } catch (fehler_try) {
                                      log('Fehler beim Kalenderevent einlesen ' + fehler_try);
                                  }
                              }</inhalt.length;>` 
                              
                              und unter Objekte bei Kalendereignisse - heute steht hat Geburtstag...
                              
                              Grüße, Rob[/i][/i][/i]
                              
                              1 Reply Last reply Reply Quote 0
                              • C
                                csamaggi last edited by

                                Seit einiger Zeit läuft mein Müllscript nicht mehr ich hatte es mir von hier kopiert.

                                `/* Kalenderevent in der Zukunft suchen
                                
                                sucht im iCal Adapter nach events, die noch kommen
                                http://forum.iobroker.net/viewtopic.php?f=21&t=3761&sid=4319378b32a0ce096bbbbfc0ebd859ce#p34975
                                
                                20.09.2016 erstellt von pix
                                13.11.2016 angepasst für Sven
                                */
                                
                                // User Anpassungen
                                var suchstring =   'Grünabfall';              // Suchbegriff
                                var stichtag =      1;                   // 0 heute;  1 morgen; 2 übermorgen; 3 In 3 Tagen
                                var idTabelle =    'ical.0.data.table'; // Instanz eintragen
                                var logging =      true;                // debug Log ein/ausschalten
                                // Ende User Anpassungen
                                
                                createState('Muellwarnung.' + suchstring, false, {
                                     type: 'boolean',
                                     name: suchstring + ' wird morgen abgeholt',
                                     desc: suchstring + ' wird morgen abgeholt (Boolean)',
                                     def: false
                                });
                                var idEventState = 'Muellwarnung.' + suchstring;
                                setState(idEventState, false); 
                                
                                var tage = ['Heute','Morgen','Übermorgen','In 3 Tagen']; // dafür relative Datumsangabe in iCal-Adapter einschalten und ggf. übersetzen
                                
                                function datum(x) {
                                    var jetzt = new Date();
                                    if (logging) log('Jetztzeit: ' + jetzt);
                                    //var zeit = new Date(jetzt.getTime() + x * 24 * 60 * 60 * 1000); // es werden genau x Tage, als x * 24 Stunden zugezählt --> nicht gut
                                    var zeit = new Date (jetzt.getFullYear(), jetzt.getMonth(), jetzt.getDate() + x); // in drei Tagen um Mitternacht
                                    if (logging) log('Zielzeit: ' + zeit);
                                    var jahr       = zeit.getFullYear();
                                    var monat      = (zeit.getMonth()+1 < 10) ? '0' + (zeit.getMonth()+1) : zeit.getMonth()+1;
                                    var tag        = (zeit.getDate() < 10) ? '0' + zeit.getDate() : zeit.getDate();
                                    return (tag + '.' + monat + '.' + jahr);
                                }
                                
                                function pruefeKalender() {
                                    var kalender = getState(idTabelle).val;
                                    var tag = datum(parseInt(stichtag,10));
                                    if (logging) log(tag);
                                    try{
                                        var ereignisse = JSON.stringify(kalender, null, 2); // Ausgabe als String
                                        if (logging) log(ereignisse);
                                        for(var i = 0; i <kalender.length; 18/i++)/{/alle/events/durchgehen/if/(/(kalender[i].date.indexof(tag)/!="-1)" ||/(kalender[i].date.indexof(tage[stichtag])/)/string/datum/oder/relatives/(nicht/nicht)/gefunden,/also/gefunden/var/termin="kalender[i].event;" (termin.indexof(suchstring)/log('treffer:/'/+/tage[stichtag]/suchstring);/hier/wird/später/vielleicht/stichtag_array/verwendet/setstate(ideventstate,/true);/}/else/false);/catch/(fehler_try)/log('fehler/beim/kalenderevents/einlesen/fehler_try,/'error');/schedule("05/*/*",/function(){/bei/mehreren/müllskripts/den/ersten/wert/verändern/(minuten)/event="getState(idEventState).val;" (event)/ansage="suchstring" morgen/abholt./bitte/rausstellen!';/kann/sayit/push/gesetzt/werden/setstate("sayit.0.tts.text"/*text/to/speech*/,/ansage);/});/aktualisierung/on/({id:/idtabelle,/change:/'any'},/function(data)/pruefekalender();/1min/nach/mitternacht/schedule("1/pruefekalender);/skriptstart/pruefekalender();<e=""></kalender.length;>`
                                
                                Es wird der Wert des State nicht mehr auf true gesetzt und somit wird mit halt auf meinem  Wandtab nichts angezeigt.
                                
                                Hoffe ihr habt da einen tipp.
                                
                                MFG Maggi[/i][/i][/i]
                                
                                1 Reply Last reply Reply Quote 0
                                • S
                                  skorpil last edited by

                                  @RobS

                                  Spontan sehe ich nichts…Frage: updates? Hast Du Adapter geupdated (was für ein Wort...). Oder am Kalender was geändert? Ich meine die Inhalte des Kalenders? Sonst habe ich leider auch keine Idee.

                                  @csamaggi

                                  Auch hier sehe ich nichts. Wenn es gelaufen ist, und Du nichts geändert hast, ist es unerklärlich. Auch hier die Frage nach Adapter updates usw.

                                  Tipp noch für Euch, wenn ihr es als Profis nicht ohnehin schon macht: Testfile einrichten mit ganz vielen Log Ausgaben und dann Schritt für Schritt den Fehler suchen. Und wenn ihr ihn habt: nicht vergessen, hier zu posten..

                                  Liebe Grüße

                                  Bernd

                                  PS: ich habe momentan so ein Problem mit der Ansage der Texte über Sonos. Ich vermute, dass das aber an dem Sonos Update liegt (hat nix mit iobroker zu tun...) bin da noch am forschen.

                                  1 Reply Last reply Reply Quote 0
                                  • C
                                    csamaggi last edited by

                                    Der Adapter ist aktuell also der JavaScript. Sonst habe ich auch nichts verändert da es ja lief.

                                    Gesendet von meinem SM-G900F mit Tapatalk

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

                                      @skorpil:

                                      Pix,

                                      Frage bzw. Hinweis:

                                      1. das mit object object habe ich manchmal, aber nicht immer. Meist stehen Werte drin. Die sich dann ja gottlob auch weiter verarbeiten lassen. Denn sonst würde mein Script ja nicht funktionieren. Das tut es aber. Eine Logik, wann ja, wann nein habe ich noch nicht ausmachen können. Manchmal ist es nur eine Frage der Zeit. Dann steht was drin…

                                      2. wenn wir also mit object immer wieder Probleme haben: warum lösen wir das nicht über html. Meine Programmierfähigkeiten reichen nicht aus, um das zu lösen. Tempestas hat das ja auch schon angeregt. Könntest Du Dich nicht erbarmen und das Script so umschreiben, dass wir einfach auf html zugreifen?

                                      Liebe Grüße

                                      Bernd `

                                      Lieber Bernd, lieber Pix,

                                      ich freue mich, jetzt (hoffentlich) auch mal mehr als nur eine Frage beitragen zu können. Ich habe das Skript von euch so abgewandelt, dass es nicht aus dem Data Object Punkt eingelesen wird sondern aus dem (immer befüllten) ical.html Teil

                                      /* ANSAGESKRIPT MIT DATUM, UHRZEIT, TEMPERATUR UND TERMINEN DES HEUTIGEN TAGES
                                      ZIELSETZUNG:  TRIGGERBAR OB NUR HEUTE; HEUTE UND MORGEN ODER BIS EINSCHLIESSLICH UEBERMORGEN
                                      ORIGNARES SKRIPT VON SKORPIL UND PIX, IOBROKER FORUM
                                      ABGEWANDELT ZWECKS AUSLESEN DER TERMINE AUS HTML ANSTATT AUS DATA OBJECT ZUR VERMEIDUNG DER OBJECT PROBLEMATIK (TERMINE WERDEN NICHT IMMER EINGELESEN)
                                      HTML TO PLAIN TEXT AUS STACKOVERFLOW FORUM; ALL CREDIT TO USER ELENDURWEN
                                      ERGAENZT DURCH TEMPESTAS ZUM AUSLESEN NUR HEUTIGER TERMINE
                                      */
                                      // Trigger: Bewegungsmelder Variable aus Homematic;
                                      var SayVar = "hm-rega.0.11430"; 
                                      //Trigger
                                          on(SayVar, function (data) {
                                      
                                          // Definition
                                          var tts = "sayit.1.tts.text";
                                          var ist = getState(SayVar).val;
                                      // logs
                                          log('SayVar:   ' + SayVar);
                                          log('TTS:   ' + tts);
                                          log('ist:   ' + ist);
                                          //Lautstärke einstellen
                                              var vol ="sayit.1.tts.volume";
                                              setState (vol,15);
                                          //Wochentag ermitteln
                                              var d = new Date ();
                                              var weekday = new Array("Sonntag","Montag","Dienstag","Mittwoch","Donnerstag","Freitag","Samstag");
                                              var w = weekday[d.getDay()]; 
                                          //Tagesdatum ermitteln
                                              var t = d.getDate();
                                          //Monat ermitteln
                                              var month = new Array("Januar","Februar","März","April","Mai","Juni","Juli","August","September","Oktober","November","Dezember");
                                              var m = month[d.getMonth()];
                                          //Jahr ermitteln
                                              var j = d.getFullYear();
                                          //Stunde ermitteln
                                              h = d.getHours();
                                          //Minute ermitteln
                                              mi = d.getMinutes();
                                          // ********************************************************************
                                          // Die Außentemperatur ist xx.
                                          // Einfache Temperaturansage mit SayIt.
                                          // Variante 1 mittels splitten der Temperatur, damit die Ansage nicht
                                          // "Es sind 18 Punkt 2 Grad " sagt.
                                          // *********************************************************************
                                              var Temperatursensor = getState("hm-rpc.0.NEQ1381269.1.TEMPERATURE"); /*Temperatursensor:1.TEMPERATURE*/
                                              var temperatur = Temperatursensor.val.toString();
                                              var temp_array = [];
                                              temp_array = temperatur.split(".");
                                                  // Fange leere Nachkommastellen ab. Das passiert, wenn die Temperatur z. B. 18.0 ist.
                                                  // Es wird dann nur "18" gelesen.
                                                  if (!temp_array[1]) {
                                                      temp_array[1] = "0";
                                                      log("Die Nach-Kommastelle in temp_array[1] war nicht vorhanden und wird nun fest auf 0 gesetzt.");
                                                }
                                      
                                          // **************************************************************************************            
                                          // Termine auswerten aus html. Bereinigung der HTML Tags und Konvertierung in Plain Text
                                          // **************************************************************************************
                                      
                                              var inhalt = getState("ical.0.data.html"/*HTML iCal table*/);
                                              var inhaltString = inhalt.val.toString();
                                              var inhaltStringReplace = inhaltString;
                                              var inhaltStringText;
                                              // remove all inside SCRIPT and STYLE tags
                                              inhaltStringReplace=inhaltStringReplace.replace(/<script.*>[\w\W]{1,}(.*?)[\w\W]{1,}<\/script>/gi, "");
                                              inhaltStringReplace=inhaltStringReplace.replace(/<style.*>[\w\W]{1,}(.*?)[\w\W]{1,}<\/style>/gi, "");
                                      
                                              // remove BR tags. Werden durch einen Punkt und Freizeichen ersetzt für sauberere Sprachausgabe
                                              inhaltStringReplace=inhaltStringReplace.replace(/
                                      /gi, ". ");
                                              inhaltStringReplace=inhaltStringReplace.replace(/<br\s\>/gi, ". ");
                                              inhaltStringReplace=inhaltStringReplace.replace(/<br\>/gi, ". ");
                                      
                                              // remove all else
                                              inhaltStringReplace=inhaltStringReplace.replace(/<(?:.|\s)*?>/g, "");
                                      
                                              // get rid of html-encoded characters:
                                              inhaltStringReplace=inhaltStringReplace.replace(/ /gi," ");
                                              inhaltStringReplace=inhaltStringReplace.replace(/&/gi,"&");
                                              inhaltStringReplace=inhaltStringReplace.replace(/"/gi,'"');
                                              inhaltStringReplace=inhaltStringReplace.replace(//gi,'>');
                                      
                                              // Punkt ans Ende setzen
                                              inhaltStringReplace=inhaltStringReplace+".";
                                      
                                              // Termine heute zaehlen
                                              var terminHeuteCount = "ical.0.data.count";
                                      
                                              //*******************************************
                                              // Text kürzen auf nur heutige Termine
                                              // ******************************************
                                      
                                              if (terminHeuteCount === 0) { 
                                                  inhaltStringText ="Es liegen keine Termine an";
                                              }
                                              else {
                                                  var n = inhaltStringReplace.indexOf("Morgen" || "Übermorgen" || "In einer Woche");
                                                  inhaltStringText = "Folgende Termine stehen an: "+ inhaltStringReplace.slice(0,n);
                                                  }    
                                              
                                      
                                          // logs
                                              log(inhaltStringText);
                                      
                                          // ANSAGE ausfuehren, danach sonos box starten für 30 Minuten
                                      
                                          if (data.newState.val === true) {
                                              setState (tts, "Guten Morgen, heute ist " + w + " der " + t + "te " + m + j + ". Es ist" + h + "  Uhr und " + mi + "  Minuten." 
                                               + "  Die Aussentemperatur beträgt " + temp_array[0] + "," + temp_array[1] + " Grad. " +inhaltStringText );
                                          setStateDelayed("sonos.0.root.192_168_2_7.favorites_set"/*favorites_set*/,'1000 Oldies', 35000);    // nach 35 Sekunden wird 1000 Oldies eingestellt
                                          setState("sonos.0.root.192_168_2_7.volume"/*volume*/, 12);
                                          setStateDelayed("sonos.0.root.192_168_2_7.state", "stop", 1860000);                                 // nach 30 Minuten wird ausgeschaltet
                                          }
                                      
                                      });</br\></br\s\></style.*></script.*>
                                      

                                      Das Skript funktioniert bei mir und auch das kürzen nur auf heutige Termine wurde durchgeführt. Eventuell kann es Probleme geben, wenn es weder weitere Termine morgen, übermorgen oder in einer Woche gibt? Das habe ich noch nicht getestet. Müsste aber glaube ich zu Fehlern führen. Ich versuche mich mal daran.

                                      Eigentlich wollte ich es so gestalten, dass man per Trigger festlegen kann, ob man nur heutige Termine, oder auch heute und morgen oder sogar bis Übermorgen hören möchte. Das Skript schießt aber immer die Javascript Instanz ab. Ich denke es liegt ab der Suchroutine für die "Morgen" bzw Übermorgen Termine. Eventuell hat da ja einer der JS Kenner eine Idee?

                                      // Variablen deklarieren        
                                      
                                              var tts = "sayit.1.tts.text";
                                              var inhalt = getState("ical.0.data.html"/*HTML iCal table*/);
                                              var inhaltString = inhalt.val.toString();
                                              var inhaltStringReplace = inhaltString;
                                              var inhaltStringText;
                                      
                                              //*******************************************************************************************************
                                              // Festlegen, ob nur heute [1], heute und morgen [2] oder einschliesslich uebermorgen angesagt wird [3]
                                              //*******************************************************************************************************
                                              var terminAnsage = 1; // hier festlegen 
                                              // remove all inside SCRIPT and STYLE tags
                                              inhaltStringReplace=inhaltStringReplace.replace(/<script.*>[\w\W]{1,}(.*?)[\w\W]{1,}<\/script>/gi, "");
                                              inhaltStringReplace=inhaltStringReplace.replace(/<style.*>[\w\W]{1,}(.*?)[\w\W]{1,}<\/style>/gi, "");
                                      
                                              // remove BR tags. Werden durch einen Punkt und Freizeichen ersetzt für sauberere Sprachausgabe
                                              inhaltStringReplace=inhaltStringReplace.replace(/
                                      /gi, ". ");
                                              inhaltStringReplace=inhaltStringReplace.replace(/<br\s\>/gi, ". ");
                                              inhaltStringReplace=inhaltStringReplace.replace(/<br\>/gi, ". ");
                                      
                                              // remove all else
                                              inhaltStringReplace=inhaltStringReplace.replace(/<(?:.|\s)*?>/g, "");
                                      
                                              // get rid of html-encoded characters:
                                              inhaltStringReplace=inhaltStringReplace.replace(/ /gi," ");
                                              inhaltStringReplace=inhaltStringReplace.replace(/&/gi,"&");
                                              inhaltStringReplace=inhaltStringReplace.replace(/"/gi,'"');
                                              inhaltStringReplace=inhaltStringReplace.replace(//gi,'>');
                                      
                                              // Punkt ans Ende setzen
                                              inhaltStringReplace=inhaltStringReplace+".";
                                      
                                              // Termine heute zaehlen
                                              var terminHeuteCount = "ical.0.data.count";
                                      
                                           /*   // Termine morgen zaehlen
                                              var i = 0;
                                              var terminMorgenCount = 0;
                                              while(i<inhaltstringreplace.length -1)/{/if("morgen"="=" inhaltstringreplace.charat(terminmorgencount))/terminmorgencount++;/}/termine/uebermorgen/zaehlen/var/k="0;" terminuebermorgencount="0;" while(k<inhaltstringreplace.length/if("Übermorgen"="=" inhaltstringreplace.charat(terminuebermorgencount))/terminuebermorgencount++;/addieren/terminecount="terminHeuteCount" +/terminmorgencount/terminuebermorgencount;/terminezukunft="terminMorgenCount" log(inhaltstringtext);/*/*******************************************/text/kürzen/je/nach/ausgewähltem/zeitraum/******************************************/if/(terminecount="==" 0)/inhaltstringtext="Es liegen keine Termine an" ;/else/switch(terminansage)/case/1:/if(terminheutecount="==" if(terminezukunft="==" n="inhaltStringReplace.indexOf(&quot;Morgen&quot;" ||/"Übermorgen");/break;/2:/if(terminuebermorgencount="==" 3:/log("insgesamt/"+terminecount/+"/in/den/nächsten/tagen");<e=""></inhaltstringreplace.length></br\></br\s\></style.*></script.*>
                                      

                                      Beste Grüße

                                      Steffen

                                      1 Reply Last reply Reply Quote 0
                                      • S
                                        skorpil last edited by

                                        Wow! Stark. Danke!

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

                                          ****UPDATE:

                                          Bitte im nachfolgenden Posting fürs gesamte Skript, jetzt auch mit Auswahl der Ansagen nur für heute oder heute und morgen oder bis einschließlich übermorgen, ansehen und ggf. dort weiter diskutieren****

                                          Moin moin,

                                          ich habe gerade festgestellt, die indexOf Variante arbeitet nicht, wenn es morgen keinen Termin gibt. Außerdem gab es einen Fehler mit dem Zählen der Termine am heutigen Tag. Hier nun das Ergebnis meiner Bemühungen heute morgen, die zu funktionieren scheinen, auch wenn es keinen Termin morgen gibt.

                                           // **************************************************************************************            
                                              // Termine auswerten aus html. Bereinigung der HTML Tags und Konvertierung in Plain Text
                                              // **************************************************************************************
                                          
                                                  var inhalt = getState("ical.0.data.html"/*HTML iCal table*/);
                                                  var inhaltString = inhalt.val.toString();
                                                  var inhaltStringReplace = inhaltString;
                                                  var inhaltStringText;
                                                  var i_search;
                                                  //**************************************************************************************
                                                  // Suchfunktion für Termin-Cutoff
                                                  // Sucht nach dem n-ten definierten Muster, hier "§$%" und gibt die Fundstelle zurück. 
                                                  // Hinter dieser Fundstelle wird dann der Text gekürzt
                                                  // Sinnloses Muster genommen, da dies wohl nirgends normalerweise vorkommt
                                                  //*****************************************************************************************
                                          
                                                  function nthIndex(str, pat, n){
                                                  var L= str.length, i= -1;
                                                  while(n-- && i++<l){ i="str.indexOf(pat," i);/if/(i/</0)/break;/}/i_search="i;" remove/all/inside/script/and/style/tags/inhaltstringreplace="inhaltStringReplace.replace(/<script.*">[\w\W]{1,}(.*?)[\w\W]{1,}<\/script>/gi, "");
                                                  inhaltStringReplace=inhaltStringReplace.replace(/<style.*>[\w\W]{1,}(.*?)[\w\W]{1,}<\/style>/gi, "");
                                          
                                                  // remove BR tags. Werden durch sinnlose Zeichenkette ersetzt, nach der später gesucht wird. 
                                                  // Es muss immer nach diesem Muster gesucht werden, da es am Ende des Termins steht. Alles rechts davon wird abgeschnitten
                                                  inhaltStringReplace=inhaltStringReplace.replace(/
                                          /gi, ". §$%");
                                                  inhaltStringReplace=inhaltStringReplace.replace(/<br\s\>/gi, ". §$%");
                                                  inhaltStringReplace=inhaltStringReplace.replace(/<br\>/gi, ". §$%");
                                          
                                                  // remove all else
                                                  inhaltStringReplace=inhaltStringReplace.replace(/<(?:.|\s)*?>/g, "");
                                          
                                                  // get rid of html-encoded characters:
                                                  inhaltStringReplace=inhaltStringReplace.replace(/ /gi," ");
                                                  inhaltStringReplace=inhaltStringReplace.replace(/&/gi,"&");
                                                  inhaltStringReplace=inhaltStringReplace.replace(/"/gi,'"');
                                                  inhaltStringReplace=inhaltStringReplace.replace(//gi,'>');
                                          
                                                  // Punkt ans Ende setzen
                                                  inhaltStringReplace=inhaltStringReplace+".";
                                          
                                                  log("Termine ohne HTML: "+inhaltStringReplace);
                                          
                                                  // Termine heute zaehlen
                                                  var terminHeuteCount = inhaltStringReplace.split("Heute").length -1;
                                                  log("Anzahl Termine heute: "+terminHeuteCount);
                                          
                                                  //*******************************************
                                                  // Text kürzen auf nur heutige Termine
                                                  // ******************************************
                                          
                                                  if (terminHeuteCount === 0) { 
                                                      inhaltStringText ="Es liegen keine Termine an";
                                          
                                                  }
                                                  else { 
                                                      nthIndex(inhaltStringReplace,"§$%", terminHeuteCount);              // nutzen der Suchfunktion zum Suchen der n-ten (entspricht Anzahl der Termine "Heute" sinnlosen Zeichenkette
                                                      inhaltStringText = inhaltStringReplace.slice(0,i_search);           // alles rechts der n-ten Fundstelle wird abgeschnitten
                                                      inhaltStringText = inhaltStringText.replace("§$%", "");             // rausnehmen der sinnlosen Zeichen, damit diese nicht mitgesprochen werden
                                                     }
                                               log(inhaltStringText);</br\></br\s\></style.*></l){> 
                                          
                                          1 Reply Last reply Reply Quote 0
                                          • T
                                            tempestas last edited by Jey Cee

                                            So, hier nun das hoffentlich finale gesamte Skript. Ich habe diverse Varianten getestet, es sollte funktionieren.

                                            Mit Sicherheit ist es nicht schön und/oder sauber geschrieben, dafür habe ich zu wenig Ahnung. Aufräumhinweise nehme ich gerne an.

                                            Wichtig: das Skript funktioniert nur für deutsche ical Einstellungen und ihr müsst die Funktion "Ersetze Datum mit Worten" aktiviert haben.

                                            Ebenfalls darf NICHT in euren Terminen selbst eins der Wörter "Heute", "Morgen" oder "Übermorgen" enthalten sein, da dies die Triggerwörter für die Suche sind aus dem ical selbst.

                                            Hier das GESAMTE SKRIPT inklusive Datum und Wetter von Skorpil und Pix. Wer nur den "neuen Teil" möchte siehe nächster Spoiler

                                            /* ANSAGESKRIPT MIT DATUM, UHRZEIT, TEMPERATUR UND TERMINEN. TRIGGERBAR. OB HEUTE, HEUTE UND MORGEN ODER BIS EINSCHLIESSLICH UEBERMORGEN
                                            ORIGNARES SKRIPT FÜR TEMPERATUR UND DATUM PLUS UHRZEIT VON SKORPIL UND PIX, IOBROKER FORUM
                                            ERGAENZT UM AUSLESEN DER TERMINE AUS HTML ANSTATT AUS DATA OBJECT ZUR VERMEIDUNG DER OBJECT PROBLEMATIK (TERMINE WERDEN NICHT IMMER EINGELESEN)
                                            HTML TO PLAIN TEXT KONVERTIERUNG AUS STACKOVERFLOW FORUM; ALL CREDIT TO USER ELENDURWEN
                                            SUCHFUNKTION STACKOVERFLOW FORUM USER iammatthew2
                                            ZUSAMMENFUERHUNRG UND TRIGGER FUNKTION TEMPESTAS
                                            VERSION V.02 STATUS 30\. APRIL 2017
                                            */
                                            // Trigger: Bewegungsmelder Variable aus Homematic;
                                            var SayVar = "hm-rega.0.11430"; 
                                            //Trigger
                                                on(SayVar, function (data) {
                                            
                                                // Definition
                                                var tts = "sayit.1.tts.text";
                                                var ist = getState(SayVar).val;
                                            // logs
                                                log('SayVar:   ' + SayVar);
                                                log('TTS:   ' + tts);
                                                log('ist:   ' + ist);
                                                //Lautstärke einstellen
                                                    var vol ="sayit.1.tts.volume";
                                                    setState (vol,15);
                                                //Wochentag ermitteln
                                                    var d = new Date ();
                                                    var weekday = new Array("Sonntag","Montag","Dienstag","Mittwoch","Donnerstag","Freitag","Samstag");
                                                    var w = weekday[d.getDay()]; 
                                                //Tagesdatum ermitteln
                                                    var t = d.getDate();
                                                //Monat ermitteln
                                                    var month = new Array("Januar","Februar","März","April","Mai","Juni","Juli","August","September","Oktober","November","Dezember");
                                                    var m = month[d.getMonth()];
                                                //Jahr ermitteln
                                                    var j = d.getFullYear();
                                                //Stunde ermitteln
                                                    h = d.getHours();
                                                //Minute ermitteln
                                                    mi = d.getMinutes();
                                                // ********************************************************************
                                                // Die Außentemperatur ist xx.
                                                // Einfache Temperaturansage mit SayIt.
                                                // Variante 1 mittels splitten der Temperatur, damit die Ansage nicht
                                                // "Es sind 18 Punkt 2 Grad " sagt.
                                                // *********************************************************************
                                                    var Temperatursensor = getState("hm-rpc.0.NEQ1381269.1.TEMPERATURE"); /*Temperatursensor:1.TEMPERATURE*/
                                                    var temperatur = Temperatursensor.val.toString();
                                                    var temp_array = [];
                                                    temp_array = temperatur.split(".");
                                                        // Fange leere Nachkommastellen ab. Das passiert, wenn die Temperatur z. B. 18.0 ist.
                                                        // Es wird dann nur "18" gelesen.
                                                        if (!temp_array[1]) {
                                                            temp_array[1] = "0";
                                                            log("Die Nach-Kommastelle in temp_array[1] war nicht vorhanden und wird nun fest auf 0 gesetzt.");
                                                      }
                                            
                                                // **************************************************************************************            
                                                // Termine auswerten aus html. Bereinigung der HTML Tags und Konvertierung in Plain Text
                                                // **************************************************************************************
                                            
                                                    var inhalt = getState("ical.0.data.html"/*HTML iCal table*/);
                                                    var inhaltString = inhalt.val.toString();
                                                    var inhaltStringReplace = inhaltString;
                                                    var inhaltStringText;
                                                    var i_search;
                                            
                                                    //*******************************************************************************************************
                                                    // Festlegen, ob nur heute [1], heute und morgen [2] oder einschliesslich uebermorgen angesagt wird [3]
                                                    //*******************************************************************************************************
                                                    var terminAnsage = 1; // hier festlegen         
                                            
                                                    //**************************************************************************************
                                                    // Suchfunktion für Termin-Cutoff
                                                    // Sucht nach dem n-ten definierten Muster, hier "§$%" und gibt die Fundstelle zurück. 
                                                    // Hinter dieser Fundstelle wird dann der Text gekürzt
                                                    // Sinnloses Muster genommen, da dies wohl nirgends normalerweise vorkommt
                                                    //*****************************************************************************************
                                            
                                                    // Suchfunktion für Termin-Cutoff
                                            
                                                    function nthIndex(str, pat, n){
                                                    var L= str.length, i= -1;
                                                    while(n-- && i++<l){ i="str.indexOf(pat," i);/if/(i/</0)/break;/}/i_search="i;" remove/all/inside/script/and/style/tags/inhaltstringreplace="inhaltStringReplace.replace(/<script.*">[\w\W]{1,}(.*?)[\w\W]{1,}<\/script>/gi, "");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/<style.*>[\w\W]{1,}(.*?)[\w\W]{1,}<\/style>/gi, "");
                                            
                                                    // remove BR tags. Werden durch sinnlose Zeichenkette ersetzt, nach der später gesucht wird
                                                    inhaltStringReplace=inhaltStringReplace.replace(/
                                            /gi, ". §$%");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/<br\s\>/gi, ". §$%");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/<br\>/gi, ". §$%");
                                            
                                                    // remove all else
                                                    inhaltStringReplace=inhaltStringReplace.replace(/<(?:.|\s)*?>/g, "");
                                            
                                                    // get rid of html-encoded characters:
                                                    inhaltStringReplace=inhaltStringReplace.replace(/ /gi," ");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/&/gi,"&");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/"/gi,'"');
                                                    inhaltStringReplace=inhaltStringReplace.replace(//gi,'>');
                                            
                                                    // Punkt ans Ende setzen
                                                    inhaltStringReplace=inhaltStringReplace+". §$%";
                                            
                                                    log("Text ohne HTML mit 'sinnlos String': "+inhaltStringReplace);
                                            
                                                    // Termine heute zaehlen
                                                    var terminHeuteCount = inhaltStringReplace.split("Heute").length -1;
                                                    log("Anzahl Termine heute: "+terminHeuteCount);
                                            
                                                    // Termine morgen zählen
                                                    var terminMorgenCount = inhaltStringReplace.split("Morgen").length -1;
                                                    log("Anzahl Termine morgen: "+terminMorgenCount);
                                            
                                                    // Termine übermorgen zählen
                                                    var terminUebermorgenCount = inhaltStringReplace.split("Übermorgen").length -1;
                                                    log("Anzahl Termine übermorgen: "+terminUebermorgenCount);
                                            
                                                    // Termine addieren
                                                    var termineCount = (terminHeuteCount + terminMorgenCount + terminUebermorgenCount);
                                                    var termineHeuteMorgenCount = (terminHeuteCount +terminMorgenCount);
                                                    log("Termine heute und morgen gesamt: " +termineHeuteMorgenCount);
                                            
                                                    //*******************************************
                                                    // Text kürzen je nach gewählter Selektion
                                                    // ******************************************
                                            
                                                    if (termineCount === 0) { 
                                                        inhaltStringText ="Es liegen keine Termine an";
                                            
                                                    }
                                                    else {
                                                        switch(terminAnsage) {
                                                            case 1:
                                                                if (terminHeuteCount === 0) { 
                                                                    inhaltStringText ="Es liegen keine Termine an";
                                                                }
                                                                else{
                                                                nthIndex(inhaltStringReplace,"§$%", terminHeuteCount);                // nutzen der Suchfunktion zum Suchen der n-ten sinnlosen Zeichenkette
                                                                inhaltStringText = inhaltStringReplace.slice(0,i_search);
                                                                for (k =0; k < terminHeuteCount; k++) {
                                                                inhaltStringText = inhaltStringText.replace("§$%", "");             // rausnehmen der sinnlosen Zeichen, damit diese nicht mitgesprochen werden
                                                                }
                                                                }
                                                            break;
                                            
                                                            case 2:
                                                                if (termineHeuteMorgenCount === 0) { 
                                                                    inhaltStringText ="Es liegen keine Termine an";
                                                                }
                                                                else{
                                                                nthIndex(inhaltStringReplace,"§$%", termineHeuteMorgenCount);            // nutzen der Suchfunktion zum Suchen der n-ten sinnlosen Zeichenkette
                                                                inhaltStringText = inhaltStringReplace.slice(0,i_search);
                                                                for (k =0; k < termineHeuteMorgenCount; k++) {
                                                                inhaltStringText = inhaltStringText.replace("§$%", "");             // rausnehmen der sinnlosen Zeichen, damit diese nicht mitgesprochen werden
                                                                }          
                                                                }                    
                                                            break; 
                                            
                                                            case 3:
                                                                if (termineCount === 0) { 
                                                                    inhaltStringText ="Es liegen keine Termine an";
                                                                }
                                                                else{
                                                                nthIndex(inhaltStringReplace,"§$%", termineCount);                // nutzen der Suchfunktion zum Suchen der n-ten sinnlosen Zeichenkette
                                                                inhaltStringText = inhaltStringReplace.slice(0,i_search);
                                                                for (k =0; k < termineCount; k++) {
                                                                inhaltStringText = inhaltStringText.replace("§$%", "");           // rausnehmen der sinnlosen Zeichen, damit diese nicht mitgesprochen werden
                                                                }
                                                                }
                                            
                                                        }
                                                    }
                                            log("Letzte Fundstelle an Position "+i_search);        
                                            log("Das Ergebnis ist: " +inhaltStringText);
                                                 
                                                // ANSAGE ausfuehren, danach sonos box starten für 30 Minuten
                                            
                                                if (data.newState.val === true) {
                                                    setState (tts, "Guten Morgen, heute ist " + w + " der " + t + "te " + m + j + ". Es ist" + h + "  Uhr und " + mi + "  Minuten." 
                                                     + "  Die Aussentemperatur beträgt " + temp_array[0] + "," + temp_array[1] + " Grad. Folgende Termine stehen an: " +inhaltStringText );
                                                setStateDelayed("sonos.0.root.192_168_2_7.favorites_set"/*favorites_set*/,'1000 Oldies', 35000);    // nach 35 Sekunden wird 1000 Oldies eingestellt
                                                setState("sonos.0.root.192_168_2_7.volume"/*volume*/, 12);
                                                setStateDelayed("sonos.0.root.192_168_2_7.state", "stop", 1860000);                                 // nach 30 Minuten wird ausgeschaltet
                                                }
                                            
                                            });</br\></br\s\></style.*></l){> 
                                            

                                            und hier nur der neue Teil:

                                                // **************************************************************************************            
                                                // Termine auswerten aus html. Bereinigung der HTML Tags und Konvertierung in Plain Text
                                                // **************************************************************************************
                                            
                                                    var inhalt = getState("ical.0.data.html"/*HTML iCal table*/);
                                                    var inhaltString = inhalt.val.toString();
                                                    var inhaltStringReplace = inhaltString;
                                                    var inhaltStringText;
                                                    var i_search;
                                            
                                                    //*******************************************************************************************************
                                                    // Festlegen, ob nur heute [1], heute und morgen [2] oder einschliesslich uebermorgen angesagt wird [3]
                                                    //*******************************************************************************************************
                                                    var terminAnsage = 1; // hier festlegen         
                                            
                                                    //**************************************************************************************
                                                    // Suchfunktion für Termin-Cutoff
                                                    // Sucht nach dem n-ten definierten Muster, hier "§$%" und gibt die Fundstelle zurück. 
                                                    // Hinter dieser Fundstelle wird dann der Text gekürzt
                                                    // Sinnloses Muster genommen, da dies wohl nirgends normalerweise vorkommt
                                                    //*****************************************************************************************
                                            
                                                    // Suchfunktion für Termin-Cutoff
                                            
                                                    function nthIndex(str, pat, n){
                                                    var L= str.length, i= -1;
                                                    while(n-- && i++<l){ i="str.indexOf(pat," i);/if/(i/</0)/break;/}/i_search="i;" remove/all/inside/script/and/style/tags/inhaltstringreplace="inhaltStringReplace.replace(/<script.*">[\w\W]{1,}(.*?)[\w\W]{1,}<\/script>/gi, "");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/<style.*>[\w\W]{1,}(.*?)[\w\W]{1,}<\/style>/gi, "");
                                            
                                                    // remove BR tags. Werden durch sinnlose Zeichenkette ersetzt, nach der später gesucht wird
                                                    inhaltStringReplace=inhaltStringReplace.replace(/
                                            /gi, ". §$%");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/<br\s\>/gi, ". §$%");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/<br\>/gi, ". §$%");
                                            
                                                    // remove all else
                                                    inhaltStringReplace=inhaltStringReplace.replace(/<(?:.|\s)*?>/g, "");
                                            
                                                    // get rid of html-encoded characters:
                                                    inhaltStringReplace=inhaltStringReplace.replace(/ /gi," ");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/&/gi,"&");
                                                    inhaltStringReplace=inhaltStringReplace.replace(/"/gi,'"');
                                                    inhaltStringReplace=inhaltStringReplace.replace(//gi,'>');
                                            
                                                    // Punkt ans Ende setzen
                                                    inhaltStringReplace=inhaltStringReplace+". §$%";
                                            
                                                    log("Text ohne HTML mit 'sinnlos String': "+inhaltStringReplace);
                                            
                                                    // Termine heute zaehlen
                                                    var terminHeuteCount = inhaltStringReplace.split("Heute").length -1;
                                                    log("Anzahl Termine heute: "+terminHeuteCount);
                                            
                                                    // Termine morgen zählen
                                                    var terminMorgenCount = inhaltStringReplace.split("Morgen").length -1;
                                                    log("Anzahl Termine morgen: "+terminMorgenCount);
                                            
                                                    // Termine übermorgen zählen
                                                    var terminUebermorgenCount = inhaltStringReplace.split("Übermorgen").length -1;
                                                    log("Anzahl Termine übermorgen: "+terminUebermorgenCount);
                                            
                                                    // Termine addieren
                                                    var termineCount = (terminHeuteCount + terminMorgenCount + terminUebermorgenCount);
                                                    var termineHeuteMorgenCount = (terminHeuteCount +terminMorgenCount);
                                                    log("Termine heute und morgen gesamt: " +termineHeuteMorgenCount);
                                            
                                                    //*******************************************
                                                    // Text kürzen je nach gewählter Selektion
                                                    // ******************************************
                                            
                                                    if (termineCount === 0) { 
                                                        inhaltStringText ="Es liegen keine Termine an";
                                            
                                                    }
                                                    else {
                                                        switch(terminAnsage) {
                                                            case 1:
                                                                if (terminHeuteCount === 0) { 
                                                                    inhaltStringText ="Es liegen keine Termine an";
                                                                }
                                                                else{
                                                                nthIndex(inhaltStringReplace,"§$%", terminHeuteCount);                // nutzen der Suchfunktion zum Suchen der n-ten sinnlosen Zeichenkette
                                                                inhaltStringText = inhaltStringReplace.slice(0,i_search);
                                                                for (k =0; k < terminHeuteCount; k++) {
                                                                inhaltStringText = inhaltStringText.replace("§$%", "");             // rausnehmen der sinnlosen Zeichen, damit diese nicht mitgesprochen werden
                                                                }
                                                                }
                                                            break;
                                            
                                                            case 2:
                                                                if (termineHeuteMorgenCount === 0) { 
                                                                    inhaltStringText ="Es liegen keine Termine an";
                                                                }
                                                                else{
                                                                nthIndex(inhaltStringReplace,"§$%", termineHeuteMorgenCount);            // nutzen der Suchfunktion zum Suchen der n-ten sinnlosen Zeichenkette
                                                                inhaltStringText = inhaltStringReplace.slice(0,i_search);
                                                                for (k =0; k < termineHeuteMorgenCount; k++) {
                                                                inhaltStringText = inhaltStringText.replace("§$%", "");             // rausnehmen der sinnlosen Zeichen, damit diese nicht mitgesprochen werden
                                                                }          
                                                                }                    
                                                            break; 
                                            
                                                            case 3:
                                                                if (termineCount === 0) { 
                                                                    inhaltStringText ="Es liegen keine Termine an";
                                                                }
                                                                else{
                                                                nthIndex(inhaltStringReplace,"§$%", termineCount);                // nutzen der Suchfunktion zum Suchen der n-ten sinnlosen Zeichenkette
                                                                inhaltStringText = inhaltStringReplace.slice(0,i_search);
                                                                for (k =0; k < termineCount; k++) {
                                                                inhaltStringText = inhaltStringText.replace("§$%", "");           // rausnehmen der sinnlosen Zeichen, damit diese nicht mitgesprochen werden
                                                                }
                                                                }
                                            
                                                        }
                                                    }
                                            log("Letzte Fundstelle an Position "+i_search);        
                                            log("Das Ergebnis ist: " +inhaltStringText);</br\></br\s\></style.*></l){> 
                                            
                                            1 Reply Last reply Reply Quote 0
                                            • First post
                                              Last post

                                            Support us

                                            ioBroker
                                            Community Adapters
                                            Donate

                                            763
                                            Online

                                            31.7k
                                            Users

                                            79.8k
                                            Topics

                                            1.3m
                                            Posts

                                            javascript
                                            26
                                            230
                                            67011
                                            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