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

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

Community Forum

  1. ioBroker Community Home
  2. Deutsch
  3. Praktische Anwendungen (Showcase)
  4. View Adapter Status

NEWS

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

  • Monatsrückblick – September 2025
    BluefoxB
    Bluefox
    13
    1
    1.9k

  • Neues Video "KI im Smart Home" - ioBroker plus n8n
    BluefoxB
    Bluefox
    15
    1
    2.2k

View Adapter Status

Geplant Angeheftet Gesperrt Verschoben Praktische Anwendungen (Showcase)
view adapterview
212 Beiträge 31 Kommentatoren 37.0k Aufrufe 52 Watching
  • Älteste zuerst
  • Neuste zuerst
  • Meiste Stimmen
Antworten
  • In einem neuen Thema antworten
Anmelden zum Antworten
Dieses Thema wurde gelöscht. Nur Nutzer mit entsprechenden Rechten können es sehen.
  • sigi234S sigi234

    @maxtor62 sagte in View Adapter Status:

    Geht das da hinten noch weiter?

    Ja

    \(BK\)\*\*<\/dt><dd>\(([0-9.]*)%\)
    \(C\)\*\*<\/dt><dd>\(([0-9.]*)%\)
    \(M\)\*\*<\/dt><dd>\(([0-9.]*)%\)
    \(Y\)\*\*<\/dt><dd>\(([0-9.]*)%\)
    
    M Offline
    M Offline
    Maxtor62
    schrieb am zuletzt editiert von Maxtor62
    #198

    @sigi234
    Danke Dir. schon verrückt. Für einen Drucker braucht man zwei verschiedene Adapter :blush:

    Funktioniert bei meinem Drucker leider nicht. Habe total andere Abfragen:

    Black[^>]+.{5}(\d+)
    agen[^>]+.{5}(\d+)
    Yell[^>]+.{5}(\d+)
    Cyan[^>]+.{5}(\d+)
    
    Num = 1
    

    Wer meint er ist was, hat aufgehört was zu werden

    1 Antwort Letzte Antwort
    0
    • D Offline
      D Offline
      Darkmann2000
      schrieb am zuletzt editiert von Darkmann2000
      #199

      Hallo Zusammen,

      ich habe nach dem Urlaub mein System auf den Aktuellen Stand gebracht, und nun folgende Fehlermeldung.

      script.js.Adapter_Status: [adapterStatus] error: moment.duration(...).format is not a function, stack: TypeError: moment.duration(...).format is not a function at Object.adapterStatus (script.js.Adapter_Status:109:115) at Job.job (C:\ioBroker\node_modules\iobroker.javascript\lib\sandbox.js:1876:34) at Job.invoke (C:\ioBroker\node_modules\node-schedule\lib\Job.js:171:15) at C:\ioBroker\node_modules\node-schedule\lib\Invocation.js:268:28 at Timeout._onTimeout (C:\ioBroker\node_modules\node-schedule\lib\Invocation.js:228:7) at listOnTimeout (node:internal/timers:581:17) at processTimers (node:internal/timers:519:7)
      

      Habt ihr eine Idee was da sein kann?

      Vielen Dank im Voraus

      2f69fd4e-6ee4-4542-912d-2c72d2a41560-image.png

      Hier das Skript

      // Imports -> müssen im Javascript Adapter unter 'Zusätzliche NPM-Module' eingetragen sein
      const moment = require("moment");
      const momentDurationFormatSetup = require("moment-duration-format");
       
       
      // Skript Einstellungen *************************************************************************************************************************************************
       
      let dpList = '0_userdata.0.vis.AdapterStatus.jsonList';                     // Datenpunkt für IconList Widget (Typ: Zeichenkette (String))
      let dpAdapterRestart = '0_userdata.0.vis.AdapterStatus.restartTrigger';     // Datenpunkt für Adapter restart (Typ: Zeichenkette (String))
       
      let dpSortMode = '0_userdata.0.vis.AdapterStatus.sortMode';                 // Datenpunkt für Sortieren (Typ: Zeichenkette (String))
      let dpFilterMode = '0_userdata.0.vis.AdapterStatus.filterMode';             // Datenpunkt für Filter (Typ: Zeichenkette (String))
       
      let adminUpdatesList = 'admin.0.info.updatesList';                          // Datenpunkt Admin Adapter für verfübare Updates der Adapter
       
      const checkInterval = 30;                                                   // Interval wie oft Status der Adapter aktualisiert werden soll (in Sekunden)
       
      let sprache = 'de';                                                         // Sprache für formatierung Dauer 
      let formatierungDauer = "dd[T] hh[h] mm[m]";                                // Formatierung der Dauer -> siehe momentjs library
       
      let neustarten = true;                                                      // true: Adapter wird neugestartet, false: Adapter wird gestoppt oder gestartet
       
      let farbeAdapterAktiv = 'green';                                            // Status Bar Farbe wenn Adapter aktiv ist
      let farbeAdapterNichtAktiv = 'FireBrick';                                   // Status Bar Farbe wenn Adapter nicht aktiv ist oder Fehler vorliegt
      let farbeAdapterDeaktiviert = 'darkgrey';                                   // Status Bar Farbe wenn Adapter deaktiviert ist
      let farbeAdapterNichtVerbunden = 'yellow';                                  // Status Bar Farbe wenn Adapter nicht verbunden ist
      let farbeAdapterZeitgesteuert = 'lightgreen';                               // Status Bar Farbe wenn Adapter zeitgesteuert ist
      let farbeAdapterErweiterung = '#44739e';                                    // Status Bar Farbe wenn Adapter Erweiterung ist
      let farbeAdapterSystem = '#44739e';                                         // Status Bar Farbe wenn Adapter mit System gestartet wird
       
      let sortResetAfter = 120;                                                   // Sortierung nach X Sekunden auf sortReset zurücksetzen (0=deaktiviert)
      let sortReset = 'memHeapUsed'                                               // Sortierung auf die zurückgesetzt werden soll
       
      let filterResetAfter = 120;                                                 // Filter nach X Sekunden zurücksetzen (0=deaktiviert)
       
      // **********************************************************************************************************************************************************************
       
       
      // Fomate für moment Lib
      moment.locale(sprache);
       
      // auf .alive Änderungen hören
      let aliveSelector = `[id=system.adapter.*.alive]`;
      let adapterAliveList = $(aliveSelector);
      if (adapterAliveList.length === 0) {
          // Fehlermeldung ausgeben, wenn selector kein result liefert
          console.error(`no result for selector '${aliveSelector}'`)
      } else {
          // listener nur für Änderung bei alive
          adapterAliveList.on(adapterStatus);
      }
       
      // auf .connection Änderungen hören
      let connectionSelector = `[id=*.info.connection]`;
      let adapterConnectionList = $(connectionSelector);
      if (adapterConnectionList.length === 0) {
          // Fehlermeldung ausgeben, wenn selector kein result liefert
          console.error(`no result for selector '${connectionSelector}'`)
      } else {
          // listener nur für Änderung bei alive
          adapterConnectionList.on(adapterStatus);
      }
       
      // auf .connected Änderungen hören
      let connectedSelector = `[id=system.adapter.*.connected]`;
      let adapterConnectedList = $(connectedSelector);
      if (adapterConnectedList.length === 0) {
          // Fehlermeldung ausgeben, wenn selector kein result liefert
          console.error(`no result for selector '${connectedSelector}'`)
      } else {
          // listener nur für Änderung bei alive
          adapterConnectedList.on(adapterStatus);
      }
       
      // auf Änderungen der Sortieung hören
      on({ id: dpSortMode, change: 'any' }, adapterStatus);
      on({ id: dpSortMode, change: 'any' }, resetSort);
       
      // auf Änderungen der Filter hören
      on({ id: dpFilterMode, change: 'any' }, adapterStatus);
      on({ id: dpFilterMode, change: 'any' }, resetFilter);
       
      // Funktion adapterStatus alle x Sekunden ausführen
      schedule('*/' + checkInterval + ' * * * * *', adapterStatus);
       
      function adapterStatus() {
          // Funktion um Status der Adapter abzurufen und als JSON String für das Material Design Widget IconList aufbereiten
          try {
              let adapterList = [];
              let updateList = myHelper().getStateValueIfExist(adminUpdatesList);
       
              for (var i = 0; i <= adapterAliveList.length - 1; i++) {
                  let id = adapterAliveList[i].replace('.alive', '');
                  let obj = getObject(adapterAliveList[i].replace('.alive', ''));
       
                  let nameArray = id.replace('system.adapter.', '').split(".");
       
                  let name = nameArray[0];
                  name = name.charAt(0).toUpperCase() + name.slice(1);
       
                  let nameWithInstance = name;
                  let adapterInstance = nameArray[1];
                  if (parseInt(adapterInstance) > 0) {
                      nameWithInstance = name + '.' + adapterInstance;
                  }
       
                  let uptime = (existsState(id + '.uptime')) ? moment.duration(getState(id + '.uptime').val, 'seconds').format(formatierungDauer, 0) : '-';
                  let image = (myHelper().checkCommonPropertyExist(obj, 'icon')) ? `/${nameArray[0]}.admin/${obj.common.icon}` : 'image-off-outline';
       
                  let hasUpdates = updateList && updateList.includes(nameArray[0]);
                  let newVersion = (hasUpdates) ? '<span class="mdi mdi-update" style="color: #ec0909;"></span>' : '';
       
                  let text = `<div style="display: flex; flex-direction: row; line-height: 1.5; padding-right: 8px; align-items: center;">
                                  <div style="flex: 1">${newVersion} ${nameWithInstance}</div>
                                  <div style="color: grey; font-size: 12px; font-family: RobotoCondensed-LightItalic;">${myHelper().getCommonPropertyIfExist(obj, 'version', '-', 'v', '')}</div>
                              </div>`
       
                  let subText = `<div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                      <div style="flex: 1;">CPU</div>
                                      <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getStateValueIfExist(id + '.cpu', '-', '', ' %')}</div>
                                  </div>
                                  <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                      <div style="flex: 1;">RAM total</div>
                                      <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getStateValueIfExist(id + '.memHeapTotal', '-', '', ' MB')}</div>
                                  </div>
                                  <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                      <div style="flex: 1;">RAM verwendet</div>
                                      <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getStateValueIfExist(id + '.memHeapUsed', '-', '', ' MB')}</div>
                                  </div>
                                  <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                      <div style="flex: 1;">RAM reserviert</div>
                                      <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getStateValueIfExist(id + '.memRss', '-', '', ' MB')}</div>
                                  </div>
                                  <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                      <div style="flex: 1;">Betriebszeit</div>
                                      <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${uptime}</div>
                                  </div>
                                  <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                      <div style="flex: 1;">Modus</div>
                                      <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getCommonPropertyIfExist(obj, 'mode', '-')}</div>
                                  </div>`
       
                  let statusBarColor = farbeAdapterNichtAktiv;
                  let status = 3;
       
                  if (myHelper().getStateValueIfExist(adapterAliveList[i]) === 'true') {
                      statusBarColor = farbeAdapterAktiv;
                      status = 0;
       
                      if (existsState(id.replace('system.adapter.', '') + '.info.connection')) {
                          if (!getState(id.replace('system.adapter.', '') + '.info.connection').val) {
                              statusBarColor = farbeAdapterNichtVerbunden;
                              status = 4;
                          }
                      } else {
                          if (myHelper().getStateValueIfExist(adapterAliveList[i].replace('.alive', '.connected')) === 'false') {
                              statusBarColor = farbeAdapterNichtVerbunden;
                              status = 4;
                          }
                      }
                  }
       
                  if (myHelper().getCommonPropertyIfExist(obj, 'mode') === 'schedule') {
                      // Adapter ist zeitgesteuert
                      statusBarColor = farbeAdapterZeitgesteuert;
                      status = 1;
                  }
       
                  if (myHelper().getCommonPropertyIfExist(obj, 'mode') === 'extension') {
                      // Adapter ist Extension
                      statusBarColor = farbeAdapterErweiterung;
                      status = 1;
                  }
       
                  if (myHelper().getCommonPropertyIfExist(obj, 'mode') === 'once') {
                      // Adapter wird mit System gestartet
                      statusBarColor = farbeAdapterSystem;
                      status = 1;
                  }
       
                  if (myHelper().getCommonPropertyIfExist(obj, 'enabled', false).toString() === 'false') {
                      // Adapter ist deaktiviert
                      statusBarColor = farbeAdapterDeaktiviert;
                      status = 2;
                  }
       
                  adapterList.push({
                      text: text,
                      subText: subText,
                      image: image,
                      listType: "buttonState",
                      objectId: dpAdapterRestart,
                      buttonStateValue: id,
                      statusBarColor: statusBarColor,
                      showValueLabel: false,
                      name: name,
                      mode: myHelper().getCommonPropertyIfExist(obj, 'mode'),
                      hasUpdates: hasUpdates,
                      cpu: parseFloat(myHelper().getStateValueIfExist(id + '.cpu', '0')),
                      memHeapTotal: parseFloat(myHelper().getStateValueIfExist(id + '.memHeapTotal', '0')),
                      memHeapUsed: parseFloat(myHelper().getStateValueIfExist(id + '.memHeapUsed', '0')),
                      memRss: parseFloat(myHelper().getStateValueIfExist(id + '.memRss', '0')),
                      uptime: parseFloat(myHelper().getStateValueIfExist(id + '.uptime', 0)),
                      status: status,
                      lockEnabled: true
                  })
              }
       
              let sortMode = myHelper().getStateValueIfExist(dpSortMode, 'name');
       
              if (sortMode === 'name' || sortMode === 'mode') {
                  adapterList.sort(function (a, b) {
                      return a[sortMode].toLowerCase() == b[sortMode].toLowerCase() ? 0 : +(a[sortMode].toLowerCase() > b[sortMode].toLowerCase()) || -1;
                  });
              } else if (sortMode === 'hasUpdates' || sortMode === 'cpu' || sortMode === 'memHeapTotal' || sortMode === 'memHeapUsed' || sortMode === 'memRss' || sortMode === 'uptime' || sortMode === 'status') {
                  adapterList.sort(function (a, b) {
                      return a[sortMode] == b[sortMode] ? 0 : +(a[sortMode] < b[sortMode]) || -1;
                  });
              } else {
                  // default: nach name sortieren
                  sortMode = 'name'
                  adapterList.sort(function (a, b) {
                      return a[sortMode].toLowerCase() == b[sortMode].toLowerCase() ? 0 : +(a[sortMode].toLowerCase() > b[sortMode].toLowerCase()) || -1;
                  });
              }
       
              // Filter: not connected, updates, deaktiviert, aktiviert
       
              let filterMode = myHelper().getStateValueIfExist(dpFilterMode, null);
       
              if (filterMode && filterMode !== null && filterMode !== '') {
                  if (filterMode === 'hasUpdates') {
                      adapterList = adapterList.filter(function (item) {
                          return item.hasUpdates === true;
                      });
                  } else if (filterMode === 'notConnected') {
                      adapterList = adapterList.filter(function (item) {
                          return item.status === 4;
                      });
                  } else if (filterMode === 'deactivated') {
                      adapterList = adapterList.filter(function (item) {
                          return item.status === 2;
                      });
                  } else if (filterMode === 'activated') {
                      adapterList = adapterList.filter(function (item) {
                          return item.status <= 1;
                      });
                  }
              }
       
              let result = JSON.stringify(adapterList);
              if (existsState(dpList) && getState(dpList).val !== result) {
                  setState(dpList, result, true);
              } else {
                  setState(dpList, result, true);
              }
       
          } catch (err) {
              console.error(`[adapterStatus] error: ${err.message}, stack: ${err.stack}`);
          }
      }
       
      // Beim Staren des Skriptes Adapter Status abrufen
      adapterStatus();
       
      // Funktion um Adapter zu starten / neu starten
      on({ id: dpAdapterRestart }, function (obj) {
          var adapter = getObject(obj.state.val.toString());
       
          if (neustarten) {
              if (adapter.common && adapter.common.enabled === false) {
                  // Adapter deaktiviert -> starten
                  adapter.common.enabled = true;
              }
          } else {
              if (adapter.common && adapter.common.enabled) {
                  adapter.common.enabled = !adapter.common.enabled;
              } else {
                  adapter.common.enabled = true;
              }
          }
       
          setObject(obj.state.val, adapter);
          console.log(`${obj.state.val.replace('system.adapter.', '')} neugestartet`);
      });
       
       
      function resetSort() {
          let sortMode = myHelper().getStateValueIfExist(dpSortMode, null);
       
          if (sortResetAfter > 0) {
              setTimeout(function () {
                  if (sortMode !== null && sortMode === myHelper().getStateValueIfExist(dpSortMode, null)) {
                      setState(dpSortMode, sortReset);
                  }
              }, sortResetAfter * 1000);
          }
      }
       
      function resetFilter() {
          let filterMode = myHelper().getStateValueIfExist(dpFilterMode, null);
       
          if (filterResetAfter > 0) {
              setTimeout(function () {
                  if (filterMode !== null && filterMode === myHelper().getStateValueIfExist(dpFilterMode, null)) {
                      setState(dpFilterMode, '');
                  }
              }, filterResetAfter * 1000);
          }
      }
       
      function myHelper() {
          return {
              getStateValueIfExist: function (id, nullValue = undefined, prepand = '', append = '') {
                  if (existsState(id)) {
                      return prepand + getState(id).val + append;
                  } else {
                      return nullValue;
                  }
              },
              getCommonPropertyIfExist: function (object, prop, nullValue = undefined, prepand = '', append = '') {
                  if (myHelper().checkCommonPropertyExist(object, prop)) {
                      return prepand + object.common[prop] + append;
                  } else {
                      return nullValue;
                  }
              },
              checkCommonPropertyExist: function (object, prop) {
                  if (object && object.common && object.common[prop]) {
                      return true;
                  } else {
                      return false;
                  }
              }
          }
      }
      

      9365282d-305b-4e33-b7e3-4cb3ed45ee47-image.png

      sigi234S 1 Antwort Letzte Antwort
      0
      • D Darkmann2000

        Hallo Zusammen,

        ich habe nach dem Urlaub mein System auf den Aktuellen Stand gebracht, und nun folgende Fehlermeldung.

        script.js.Adapter_Status: [adapterStatus] error: moment.duration(...).format is not a function, stack: TypeError: moment.duration(...).format is not a function at Object.adapterStatus (script.js.Adapter_Status:109:115) at Job.job (C:\ioBroker\node_modules\iobroker.javascript\lib\sandbox.js:1876:34) at Job.invoke (C:\ioBroker\node_modules\node-schedule\lib\Job.js:171:15) at C:\ioBroker\node_modules\node-schedule\lib\Invocation.js:268:28 at Timeout._onTimeout (C:\ioBroker\node_modules\node-schedule\lib\Invocation.js:228:7) at listOnTimeout (node:internal/timers:581:17) at processTimers (node:internal/timers:519:7)
        

        Habt ihr eine Idee was da sein kann?

        Vielen Dank im Voraus

        2f69fd4e-6ee4-4542-912d-2c72d2a41560-image.png

        Hier das Skript

        // Imports -> müssen im Javascript Adapter unter 'Zusätzliche NPM-Module' eingetragen sein
        const moment = require("moment");
        const momentDurationFormatSetup = require("moment-duration-format");
         
         
        // Skript Einstellungen *************************************************************************************************************************************************
         
        let dpList = '0_userdata.0.vis.AdapterStatus.jsonList';                     // Datenpunkt für IconList Widget (Typ: Zeichenkette (String))
        let dpAdapterRestart = '0_userdata.0.vis.AdapterStatus.restartTrigger';     // Datenpunkt für Adapter restart (Typ: Zeichenkette (String))
         
        let dpSortMode = '0_userdata.0.vis.AdapterStatus.sortMode';                 // Datenpunkt für Sortieren (Typ: Zeichenkette (String))
        let dpFilterMode = '0_userdata.0.vis.AdapterStatus.filterMode';             // Datenpunkt für Filter (Typ: Zeichenkette (String))
         
        let adminUpdatesList = 'admin.0.info.updatesList';                          // Datenpunkt Admin Adapter für verfübare Updates der Adapter
         
        const checkInterval = 30;                                                   // Interval wie oft Status der Adapter aktualisiert werden soll (in Sekunden)
         
        let sprache = 'de';                                                         // Sprache für formatierung Dauer 
        let formatierungDauer = "dd[T] hh[h] mm[m]";                                // Formatierung der Dauer -> siehe momentjs library
         
        let neustarten = true;                                                      // true: Adapter wird neugestartet, false: Adapter wird gestoppt oder gestartet
         
        let farbeAdapterAktiv = 'green';                                            // Status Bar Farbe wenn Adapter aktiv ist
        let farbeAdapterNichtAktiv = 'FireBrick';                                   // Status Bar Farbe wenn Adapter nicht aktiv ist oder Fehler vorliegt
        let farbeAdapterDeaktiviert = 'darkgrey';                                   // Status Bar Farbe wenn Adapter deaktiviert ist
        let farbeAdapterNichtVerbunden = 'yellow';                                  // Status Bar Farbe wenn Adapter nicht verbunden ist
        let farbeAdapterZeitgesteuert = 'lightgreen';                               // Status Bar Farbe wenn Adapter zeitgesteuert ist
        let farbeAdapterErweiterung = '#44739e';                                    // Status Bar Farbe wenn Adapter Erweiterung ist
        let farbeAdapterSystem = '#44739e';                                         // Status Bar Farbe wenn Adapter mit System gestartet wird
         
        let sortResetAfter = 120;                                                   // Sortierung nach X Sekunden auf sortReset zurücksetzen (0=deaktiviert)
        let sortReset = 'memHeapUsed'                                               // Sortierung auf die zurückgesetzt werden soll
         
        let filterResetAfter = 120;                                                 // Filter nach X Sekunden zurücksetzen (0=deaktiviert)
         
        // **********************************************************************************************************************************************************************
         
         
        // Fomate für moment Lib
        moment.locale(sprache);
         
        // auf .alive Änderungen hören
        let aliveSelector = `[id=system.adapter.*.alive]`;
        let adapterAliveList = $(aliveSelector);
        if (adapterAliveList.length === 0) {
            // Fehlermeldung ausgeben, wenn selector kein result liefert
            console.error(`no result for selector '${aliveSelector}'`)
        } else {
            // listener nur für Änderung bei alive
            adapterAliveList.on(adapterStatus);
        }
         
        // auf .connection Änderungen hören
        let connectionSelector = `[id=*.info.connection]`;
        let adapterConnectionList = $(connectionSelector);
        if (adapterConnectionList.length === 0) {
            // Fehlermeldung ausgeben, wenn selector kein result liefert
            console.error(`no result for selector '${connectionSelector}'`)
        } else {
            // listener nur für Änderung bei alive
            adapterConnectionList.on(adapterStatus);
        }
         
        // auf .connected Änderungen hören
        let connectedSelector = `[id=system.adapter.*.connected]`;
        let adapterConnectedList = $(connectedSelector);
        if (adapterConnectedList.length === 0) {
            // Fehlermeldung ausgeben, wenn selector kein result liefert
            console.error(`no result for selector '${connectedSelector}'`)
        } else {
            // listener nur für Änderung bei alive
            adapterConnectedList.on(adapterStatus);
        }
         
        // auf Änderungen der Sortieung hören
        on({ id: dpSortMode, change: 'any' }, adapterStatus);
        on({ id: dpSortMode, change: 'any' }, resetSort);
         
        // auf Änderungen der Filter hören
        on({ id: dpFilterMode, change: 'any' }, adapterStatus);
        on({ id: dpFilterMode, change: 'any' }, resetFilter);
         
        // Funktion adapterStatus alle x Sekunden ausführen
        schedule('*/' + checkInterval + ' * * * * *', adapterStatus);
         
        function adapterStatus() {
            // Funktion um Status der Adapter abzurufen und als JSON String für das Material Design Widget IconList aufbereiten
            try {
                let adapterList = [];
                let updateList = myHelper().getStateValueIfExist(adminUpdatesList);
         
                for (var i = 0; i <= adapterAliveList.length - 1; i++) {
                    let id = adapterAliveList[i].replace('.alive', '');
                    let obj = getObject(adapterAliveList[i].replace('.alive', ''));
         
                    let nameArray = id.replace('system.adapter.', '').split(".");
         
                    let name = nameArray[0];
                    name = name.charAt(0).toUpperCase() + name.slice(1);
         
                    let nameWithInstance = name;
                    let adapterInstance = nameArray[1];
                    if (parseInt(adapterInstance) > 0) {
                        nameWithInstance = name + '.' + adapterInstance;
                    }
         
                    let uptime = (existsState(id + '.uptime')) ? moment.duration(getState(id + '.uptime').val, 'seconds').format(formatierungDauer, 0) : '-';
                    let image = (myHelper().checkCommonPropertyExist(obj, 'icon')) ? `/${nameArray[0]}.admin/${obj.common.icon}` : 'image-off-outline';
         
                    let hasUpdates = updateList && updateList.includes(nameArray[0]);
                    let newVersion = (hasUpdates) ? '<span class="mdi mdi-update" style="color: #ec0909;"></span>' : '';
         
                    let text = `<div style="display: flex; flex-direction: row; line-height: 1.5; padding-right: 8px; align-items: center;">
                                    <div style="flex: 1">${newVersion} ${nameWithInstance}</div>
                                    <div style="color: grey; font-size: 12px; font-family: RobotoCondensed-LightItalic;">${myHelper().getCommonPropertyIfExist(obj, 'version', '-', 'v', '')}</div>
                                </div>`
         
                    let subText = `<div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                        <div style="flex: 1;">CPU</div>
                                        <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getStateValueIfExist(id + '.cpu', '-', '', ' %')}</div>
                                    </div>
                                    <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                        <div style="flex: 1;">RAM total</div>
                                        <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getStateValueIfExist(id + '.memHeapTotal', '-', '', ' MB')}</div>
                                    </div>
                                    <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                        <div style="flex: 1;">RAM verwendet</div>
                                        <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getStateValueIfExist(id + '.memHeapUsed', '-', '', ' MB')}</div>
                                    </div>
                                    <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                        <div style="flex: 1;">RAM reserviert</div>
                                        <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getStateValueIfExist(id + '.memRss', '-', '', ' MB')}</div>
                                    </div>
                                    <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                        <div style="flex: 1;">Betriebszeit</div>
                                        <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${uptime}</div>
                                    </div>
                                    <div style="display: flex; flex-direction: row; line-height: 1.3; padding-left: 1px; padding-right: 8px; align-items: center;">
                                        <div style="flex: 1;">Modus</div>
                                        <div style="color: grey; font-size: 14px; font-family: RobotoCondensed-LightItalic; text-align: right;">${myHelper().getCommonPropertyIfExist(obj, 'mode', '-')}</div>
                                    </div>`
         
                    let statusBarColor = farbeAdapterNichtAktiv;
                    let status = 3;
         
                    if (myHelper().getStateValueIfExist(adapterAliveList[i]) === 'true') {
                        statusBarColor = farbeAdapterAktiv;
                        status = 0;
         
                        if (existsState(id.replace('system.adapter.', '') + '.info.connection')) {
                            if (!getState(id.replace('system.adapter.', '') + '.info.connection').val) {
                                statusBarColor = farbeAdapterNichtVerbunden;
                                status = 4;
                            }
                        } else {
                            if (myHelper().getStateValueIfExist(adapterAliveList[i].replace('.alive', '.connected')) === 'false') {
                                statusBarColor = farbeAdapterNichtVerbunden;
                                status = 4;
                            }
                        }
                    }
         
                    if (myHelper().getCommonPropertyIfExist(obj, 'mode') === 'schedule') {
                        // Adapter ist zeitgesteuert
                        statusBarColor = farbeAdapterZeitgesteuert;
                        status = 1;
                    }
         
                    if (myHelper().getCommonPropertyIfExist(obj, 'mode') === 'extension') {
                        // Adapter ist Extension
                        statusBarColor = farbeAdapterErweiterung;
                        status = 1;
                    }
         
                    if (myHelper().getCommonPropertyIfExist(obj, 'mode') === 'once') {
                        // Adapter wird mit System gestartet
                        statusBarColor = farbeAdapterSystem;
                        status = 1;
                    }
         
                    if (myHelper().getCommonPropertyIfExist(obj, 'enabled', false).toString() === 'false') {
                        // Adapter ist deaktiviert
                        statusBarColor = farbeAdapterDeaktiviert;
                        status = 2;
                    }
         
                    adapterList.push({
                        text: text,
                        subText: subText,
                        image: image,
                        listType: "buttonState",
                        objectId: dpAdapterRestart,
                        buttonStateValue: id,
                        statusBarColor: statusBarColor,
                        showValueLabel: false,
                        name: name,
                        mode: myHelper().getCommonPropertyIfExist(obj, 'mode'),
                        hasUpdates: hasUpdates,
                        cpu: parseFloat(myHelper().getStateValueIfExist(id + '.cpu', '0')),
                        memHeapTotal: parseFloat(myHelper().getStateValueIfExist(id + '.memHeapTotal', '0')),
                        memHeapUsed: parseFloat(myHelper().getStateValueIfExist(id + '.memHeapUsed', '0')),
                        memRss: parseFloat(myHelper().getStateValueIfExist(id + '.memRss', '0')),
                        uptime: parseFloat(myHelper().getStateValueIfExist(id + '.uptime', 0)),
                        status: status,
                        lockEnabled: true
                    })
                }
         
                let sortMode = myHelper().getStateValueIfExist(dpSortMode, 'name');
         
                if (sortMode === 'name' || sortMode === 'mode') {
                    adapterList.sort(function (a, b) {
                        return a[sortMode].toLowerCase() == b[sortMode].toLowerCase() ? 0 : +(a[sortMode].toLowerCase() > b[sortMode].toLowerCase()) || -1;
                    });
                } else if (sortMode === 'hasUpdates' || sortMode === 'cpu' || sortMode === 'memHeapTotal' || sortMode === 'memHeapUsed' || sortMode === 'memRss' || sortMode === 'uptime' || sortMode === 'status') {
                    adapterList.sort(function (a, b) {
                        return a[sortMode] == b[sortMode] ? 0 : +(a[sortMode] < b[sortMode]) || -1;
                    });
                } else {
                    // default: nach name sortieren
                    sortMode = 'name'
                    adapterList.sort(function (a, b) {
                        return a[sortMode].toLowerCase() == b[sortMode].toLowerCase() ? 0 : +(a[sortMode].toLowerCase() > b[sortMode].toLowerCase()) || -1;
                    });
                }
         
                // Filter: not connected, updates, deaktiviert, aktiviert
         
                let filterMode = myHelper().getStateValueIfExist(dpFilterMode, null);
         
                if (filterMode && filterMode !== null && filterMode !== '') {
                    if (filterMode === 'hasUpdates') {
                        adapterList = adapterList.filter(function (item) {
                            return item.hasUpdates === true;
                        });
                    } else if (filterMode === 'notConnected') {
                        adapterList = adapterList.filter(function (item) {
                            return item.status === 4;
                        });
                    } else if (filterMode === 'deactivated') {
                        adapterList = adapterList.filter(function (item) {
                            return item.status === 2;
                        });
                    } else if (filterMode === 'activated') {
                        adapterList = adapterList.filter(function (item) {
                            return item.status <= 1;
                        });
                    }
                }
         
                let result = JSON.stringify(adapterList);
                if (existsState(dpList) && getState(dpList).val !== result) {
                    setState(dpList, result, true);
                } else {
                    setState(dpList, result, true);
                }
         
            } catch (err) {
                console.error(`[adapterStatus] error: ${err.message}, stack: ${err.stack}`);
            }
        }
         
        // Beim Staren des Skriptes Adapter Status abrufen
        adapterStatus();
         
        // Funktion um Adapter zu starten / neu starten
        on({ id: dpAdapterRestart }, function (obj) {
            var adapter = getObject(obj.state.val.toString());
         
            if (neustarten) {
                if (adapter.common && adapter.common.enabled === false) {
                    // Adapter deaktiviert -> starten
                    adapter.common.enabled = true;
                }
            } else {
                if (adapter.common && adapter.common.enabled) {
                    adapter.common.enabled = !adapter.common.enabled;
                } else {
                    adapter.common.enabled = true;
                }
            }
         
            setObject(obj.state.val, adapter);
            console.log(`${obj.state.val.replace('system.adapter.', '')} neugestartet`);
        });
         
         
        function resetSort() {
            let sortMode = myHelper().getStateValueIfExist(dpSortMode, null);
         
            if (sortResetAfter > 0) {
                setTimeout(function () {
                    if (sortMode !== null && sortMode === myHelper().getStateValueIfExist(dpSortMode, null)) {
                        setState(dpSortMode, sortReset);
                    }
                }, sortResetAfter * 1000);
            }
        }
         
        function resetFilter() {
            let filterMode = myHelper().getStateValueIfExist(dpFilterMode, null);
         
            if (filterResetAfter > 0) {
                setTimeout(function () {
                    if (filterMode !== null && filterMode === myHelper().getStateValueIfExist(dpFilterMode, null)) {
                        setState(dpFilterMode, '');
                    }
                }, filterResetAfter * 1000);
            }
        }
         
        function myHelper() {
            return {
                getStateValueIfExist: function (id, nullValue = undefined, prepand = '', append = '') {
                    if (existsState(id)) {
                        return prepand + getState(id).val + append;
                    } else {
                        return nullValue;
                    }
                },
                getCommonPropertyIfExist: function (object, prop, nullValue = undefined, prepand = '', append = '') {
                    if (myHelper().checkCommonPropertyExist(object, prop)) {
                        return prepand + object.common[prop] + append;
                    } else {
                        return nullValue;
                    }
                },
                checkCommonPropertyExist: function (object, prop) {
                    if (object && object.common && object.common[prop]) {
                        return true;
                    } else {
                        return false;
                    }
                }
            }
        }
        

        9365282d-305b-4e33-b7e3-4cb3ed45ee47-image.png

        sigi234S Online
        sigi234S Online
        sigi234
        Forum Testing Most Active
        schrieb am zuletzt editiert von
        #200

        @darkmann2000

        Lösche es mal aus dem Adapter, glaube es wird nicht mehr benötigt da es jetzt integriert ist.

        Bitte benutzt das Voting rechts unten im Beitrag wenn er euch geholfen hat.
        Immer Daten sichern!

        D 1 Antwort Letzte Antwort
        0
        • sigi234S sigi234

          @darkmann2000

          Lösche es mal aus dem Adapter, glaube es wird nicht mehr benötigt da es jetzt integriert ist.

          D Offline
          D Offline
          Darkmann2000
          schrieb am zuletzt editiert von
          #201

          @sigi234

          Hi,
          ist leider immer noch

          	script.js.Adapter_Status: [adapterStatus] error: moment.duration(...).format is not a function, stack: TypeError: moment.duration(...).format is not a function at Object.adapterStatus (script.js.Adapter_Status:109:115) at Object.callback (C:\ioBroker\node_modules\iobroker.javascript\lib\sandbox.js:1452:38) at Object.stateChange (C:\ioBroker\node_modules\iobroker.javascript\main.js:645:29) at Immediate._onImmediate (C:\ioBroker\node_modules\@iobroker\js-controller-adapter\src\lib\adapter\adapter.ts:11053:62) at processImmediate (node:internal/timers:483:21)
          

          Hier habe ich die entfernt.
          5578a63d-b7bc-46a4-9c28-c066ce9c88b6-image.png

          sigi234S 1 Antwort Letzte Antwort
          0
          • D Darkmann2000

            @sigi234

            Hi,
            ist leider immer noch

            	script.js.Adapter_Status: [adapterStatus] error: moment.duration(...).format is not a function, stack: TypeError: moment.duration(...).format is not a function at Object.adapterStatus (script.js.Adapter_Status:109:115) at Object.callback (C:\ioBroker\node_modules\iobroker.javascript\lib\sandbox.js:1452:38) at Object.stateChange (C:\ioBroker\node_modules\iobroker.javascript\main.js:645:29) at Immediate._onImmediate (C:\ioBroker\node_modules\@iobroker\js-controller-adapter\src\lib\adapter\adapter.ts:11053:62) at processImmediate (node:internal/timers:483:21)
            

            Hier habe ich die entfernt.
            5578a63d-b7bc-46a4-9c28-c066ce9c88b6-image.png

            sigi234S Online
            sigi234S Online
            sigi234
            Forum Testing Most Active
            schrieb am zuletzt editiert von sigi234
            #202

            @darkmann2000

            Denke da muss das Skript angepasst werden.

            Ev. kann da @Codierknecht helfen?

            Oder @haus-automatisierung ?

            Bitte benutzt das Voting rechts unten im Beitrag wenn er euch geholfen hat.
            Immer Daten sichern!

            D 1 Antwort Letzte Antwort
            0
            • sigi234S sigi234

              @darkmann2000

              Denke da muss das Skript angepasst werden.

              Ev. kann da @Codierknecht helfen?

              Oder @haus-automatisierung ?

              D Offline
              D Offline
              Darkmann2000
              schrieb am zuletzt editiert von
              #203

              @sigi234

              Danke für deine Hilfe.

              Das Problem sollten dann ja alle haben die den Adapter nutzen.

              1 Antwort Letzte Antwort
              0
              • F Offline
                F Offline
                Frank579
                schrieb am zuletzt editiert von
                #204

                @darkmann2000
                Pack das momentDurationFormatSetup(moment); mal in zeile 4 vom Script
                Hab ich entweder auf Git oder hier im Forum gefunden.
                Habe die Module noch im Javascript Adapter drin, weiß aber nicht obs ohne get ;)

                D 1 Antwort Letzte Antwort
                1
                • F Frank579

                  @darkmann2000
                  Pack das momentDurationFormatSetup(moment); mal in zeile 4 vom Script
                  Hab ich entweder auf Git oder hier im Forum gefunden.
                  Habe die Module noch im Javascript Adapter drin, weiß aber nicht obs ohne get ;)

                  D Offline
                  D Offline
                  Darkmann2000
                  schrieb am zuletzt editiert von Darkmann2000
                  #205

                  @frank579 sagte in View Adapter Status:

                  momentDurationFormatSetup(moment);

                  Hi,

                  jetzt funktionert es wieder.

                  PS: Habe das nur im Skript hinzugefügt.
                  Das braucht nicht unter den Zusätzlichen NPM Modulen hinzugefügt werden.

                  Vielen Dank an euch beiden für die schnelle Hilfe.

                  sigi234S 1 Antwort Letzte Antwort
                  0
                  • D Darkmann2000

                    @frank579 sagte in View Adapter Status:

                    momentDurationFormatSetup(moment);

                    Hi,

                    jetzt funktionert es wieder.

                    PS: Habe das nur im Skript hinzugefügt.
                    Das braucht nicht unter den Zusätzlichen NPM Modulen hinzugefügt werden.

                    Vielen Dank an euch beiden für die schnelle Hilfe.

                    sigi234S Online
                    sigi234S Online
                    sigi234
                    Forum Testing Most Active
                    schrieb am zuletzt editiert von
                    #206

                    @darkmann2000 sagte in View Adapter Status:

                    @frank579 sagte in View Adapter Status:

                    momentDurationFormatSetup(moment);

                    Hi,

                    jetzt funktionert es wieder.

                    PS: Habe das nur im Skript hinzugefügt.
                    Das braucht nicht unter den Zusätzlichen NPM Modulen hinzugefügt werden.

                    Vielen Dank an euch beiden für die schnelle Hilfe.

                    Zusätzliche Module sind jetzt leer bei dir?

                    Bitte benutzt das Voting rechts unten im Beitrag wenn er euch geholfen hat.
                    Immer Daten sichern!

                    L 1 Antwort Letzte Antwort
                    0
                    • D Offline
                      D Offline
                      Darkmann2000
                      schrieb am zuletzt editiert von Darkmann2000
                      #207

                      @sigi234

                      Nein, das steht noch drin.

                      Ich meinte damit

                      4655e738-8612-4b26-bc33-f1a4786f79e8-image.png

                      1 Antwort Letzte Antwort
                      0
                      • sigi234S sigi234

                        @darkmann2000 sagte in View Adapter Status:

                        @frank579 sagte in View Adapter Status:

                        momentDurationFormatSetup(moment);

                        Hi,

                        jetzt funktionert es wieder.

                        PS: Habe das nur im Skript hinzugefügt.
                        Das braucht nicht unter den Zusätzlichen NPM Modulen hinzugefügt werden.

                        Vielen Dank an euch beiden für die schnelle Hilfe.

                        Zusätzliche Module sind jetzt leer bei dir?

                        L Offline
                        L Offline
                        leonundjulie
                        schrieb am zuletzt editiert von
                        #208

                        @sigi234 guten Morgen - der frühe Vogel fängt den Wurm ;-)

                        Kann man die Anzeige in Breite und Höhe begrenzen? Ich möchte in einem View noch am linken Rand des Monitors und in einem anderen View anderen View am unteren Rand des Monitors etwas platzieren wie zB ein Menü. Was muss man da tun?

                        10x JA-Solar (380Wp) unter 15° gen Süden und 12x JA-Solar (385Wp) unter 10° gen Westen auf Carport, HUAWEI SUN2000 WR

                        Daran HUAWEI LUNA2000, 10kWh

                        Und via Kaskade 8x JA-Solar (385Wp) unter 45° gen Osten und 14x JA-Solar (385Wp) unter 23° gen Westen am 2. HUAWEI SUN2000 WR.

                        SMARTFOX PRO als EMS, KEBA P30-X Wallbox

                        R 1 Antwort Letzte Antwort
                        0
                        • L leonundjulie

                          @sigi234 guten Morgen - der frühe Vogel fängt den Wurm ;-)

                          Kann man die Anzeige in Breite und Höhe begrenzen? Ich möchte in einem View noch am linken Rand des Monitors und in einem anderen View anderen View am unteren Rand des Monitors etwas platzieren wie zB ein Menü. Was muss man da tun?

                          R Offline
                          R Offline
                          Ralf 2
                          schrieb am zuletzt editiert von
                          #209

                          @leonundjulie
                          Dann mußt du bei allen Widgets die Positionen anpassen. Ich habe links und oben meine Menüs. Z.B.: die Iconlist:
                          8c9fea94-3970-4440-a0fe-b94ce1228be1-image.png

                          Gruß Ralf

                          L 1 Antwort Letzte Antwort
                          0
                          • R Ralf 2

                            @leonundjulie
                            Dann mußt du bei allen Widgets die Positionen anpassen. Ich habe links und oben meine Menüs. Z.B.: die Iconlist:
                            8c9fea94-3970-4440-a0fe-b94ce1228be1-image.png

                            L Offline
                            L Offline
                            leonundjulie
                            schrieb am zuletzt editiert von leonundjulie
                            #210

                            @ralf-2 danke für den Tip. Kannst Du bitte ein Screenshot vom Ergebnis, also von der ganzen Monitorseite senden?

                            10x JA-Solar (380Wp) unter 15° gen Süden und 12x JA-Solar (385Wp) unter 10° gen Westen auf Carport, HUAWEI SUN2000 WR

                            Daran HUAWEI LUNA2000, 10kWh

                            Und via Kaskade 8x JA-Solar (385Wp) unter 45° gen Osten und 14x JA-Solar (385Wp) unter 23° gen Westen am 2. HUAWEI SUN2000 WR.

                            SMARTFOX PRO als EMS, KEBA P30-X Wallbox

                            R 1 Antwort Letzte Antwort
                            0
                            • L leonundjulie

                              @ralf-2 danke für den Tip. Kannst Du bitte ein Screenshot vom Ergebnis, also von der ganzen Monitorseite senden?

                              R Offline
                              R Offline
                              Ralf 2
                              schrieb am zuletzt editiert von
                              #211

                              @leonundjulie
                              Kann ich ja, was versprichst du dir davon?
                              Screenshot bei verkleinertem Browserfenster:
                              fccbf39f-c126-4ecb-82e0-3d0a41677b9f-image.png

                              Gruß Ralf

                              L 1 Antwort Letzte Antwort
                              0
                              • R Ralf 2

                                @leonundjulie
                                Kann ich ja, was versprichst du dir davon?
                                Screenshot bei verkleinertem Browserfenster:
                                fccbf39f-c126-4ecb-82e0-3d0a41677b9f-image.png

                                L Offline
                                L Offline
                                leonundjulie
                                schrieb am zuletzt editiert von
                                #212

                                @ralf-2 Danke Ralf. Ich war mir nicht sicher, ob ich Deinen Rat richtig verstand und bat Dich deshalb um das Foto. Aber letztlich habe ich es verstanden und bin am Ziel. Mir war bis dato nicht klar, dass man die Längen/Breiten Definition auch mit Prozenten angeben kann, so dass sich das ganze bei der Angabe 100% automatisch a die Bildschirmgröße anpasst … ich wollte aber feste Werte einstellen - gesagt getan, passt es jetzt zu meinem iPad und am unteren Rand habe ich mein Menü.

                                Weil ich hier nicht das richtige iPad zur Hand habe, kann ich nur dieses Foto anheften.
                                IMG_1294.jpeg

                                10x JA-Solar (380Wp) unter 15° gen Süden und 12x JA-Solar (385Wp) unter 10° gen Westen auf Carport, HUAWEI SUN2000 WR

                                Daran HUAWEI LUNA2000, 10kWh

                                Und via Kaskade 8x JA-Solar (385Wp) unter 45° gen Osten und 14x JA-Solar (385Wp) unter 23° gen Westen am 2. HUAWEI SUN2000 WR.

                                SMARTFOX PRO als EMS, KEBA P30-X Wallbox

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


                                Support us

                                ioBroker
                                Community Adapters
                                Donate

                                263

                                Online

                                32.4k

                                Benutzer

                                81.4k

                                Themen

                                1.3m

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

                                • Du hast noch kein Konto? Registrieren

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