RegisterPropertyInteger('VarProduction', 0); $this->RegisterPropertyInteger('VarConsumption', 0); $this->RegisterPropertyInteger('VarFeedIn', 0); $this->RegisterPropertyInteger('VarGrid', 0); // Persisted UI state $this->RegisterAttributeString(self::ATTR_RANGE, 'day'); $this->RegisterAttributeString(self::ATTR_DATE, date('Y-m-d')); // Enable individual visualization (HTML-SDK) $this->SetVisualizationType(1); // IMPORTANT: Timer calls global helper below (must exist!) $this->RegisterTimer('AutoPush', 0, 'IPS_RequestAction($_IPS["TARGET"], "Refresh", 1);'); } public function ApplyChanges(): void { parent::ApplyChanges(); // ensure range valid $range = $this->ReadAttributeString(self::ATTR_RANGE); if (!in_array($range, ['day', 'week', 'month', 'year', 'total'], true)) { $this->WriteAttributeString(self::ATTR_RANGE, 'day'); } // ensure date valid (not empty/invalid/future) $date = $this->ReadAttributeString(self::ATTR_DATE); if ($date === '' || !$this->isValidDate($date) || strtotime($date . ' 00:00:00') > time()) { $this->WriteAttributeString(self::ATTR_DATE, date('Y-m-d')); } // Fullscreen-Fix: push periodically (adjust as you like) // 2000ms = alle 2 Sekunden (stabil, aber nicht ganz so brutal wie 1000ms) $this->SetTimerInterval('AutoPush', 2000); $this->RecalculateAndPush(); } public function GetVisualizationTile(): string { $path = __DIR__ . '/module.html'; if (!file_exists($path)) { return '
module.html fehlt
'; } return file_get_contents($path); } public function RequestAction($Ident, $Value): void { switch ($Ident) { case 'SetRange': $range = (string)$Value; if (!in_array($range, ['day', 'week', 'month', 'year', 'total'], true)) { return; } $this->WriteAttributeString(self::ATTR_RANGE, $range); $this->RecalculateAndPush(); break; case 'SetDate': $date = (string)$Value; if (!$this->isValidDate($date)) { return; } $this->WriteAttributeString(self::ATTR_DATE, $date); $this->RecalculateAndPush(); break; case 'Prev': case 'Next': case 'Today': $this->ShiftDate($Ident); $this->RecalculateAndPush(); break; case 'Refresh': $this->RecalculateAndPush(); break; default: // ignore unknown return; } } private function RecalculateAndPush(): void { $range = $this->ReadAttributeString(self::ATTR_RANGE); $date = $this->ReadAttributeString(self::ATTR_DATE); [$tStart, $tEnd] = $this->getRange($range, $date); $dbgProd = []; $dbgFeed = []; $dbgGrid = []; $prod = $this->readDelta($this->ReadPropertyInteger('VarProduction'), $tStart, $tEnd, $dbgProd); $feed = $this->readDelta($this->ReadPropertyInteger('VarFeedIn'), $tStart, $tEnd, $dbgFeed); $grid = $this->readDelta($this->ReadPropertyInteger('VarGrid'), $tStart, $tEnd, $dbgGrid); $hasData = (($dbgProd['count'] ?? 0) > 0) || (($dbgFeed['count'] ?? 0) > 0) || (($dbgGrid['count'] ?? 0) > 0); $noDataHint = (!$hasData && $range !== 'total') ? 'Letzter Zeitpunkt' : ''; // House = Prod - Feed + Grid $house = $prod - $feed + $grid; if ($house < 0) $house = 0.0; $payload = [ 'range' => $range, 'date' => $date, 'tStart' => $tStart, 'tEnd' => $tEnd, 'hasData' => $hasData, 'noDataHint' => $noDataHint, 'values' => [ 'Produktion' => (float)$prod, 'Einspeisung' => (float)$feed, 'Netz' => (float)$grid, 'Hausverbrauch' => (float)$house ], // kannst du später entfernen 'debug' => [ 'prod' => $dbgProd, 'feed' => $dbgFeed, 'grid' => $dbgGrid ] ]; $this->UpdateVisualizationValue(json_encode($payload, JSON_THROW_ON_ERROR)); } private function getRange(string $range, string $dateYmd): array { $now = time(); if ($range === 'total') { return [0, $now]; } $base = strtotime($dateYmd . ' 00:00:00') ?: strtotime(date('Y-m-d') . ' 00:00:00'); switch ($range) { case 'day': return [$base, $base + 86400]; case 'week': $dow = (int)date('N', $base); // 1=Mon..7=Sun $start = $base - (($dow - 1) * 86400); return [$start, $start + 7 * 86400]; case 'month': $start = strtotime(date('Y-m-01 00:00:00', $base)); $end = strtotime(date('Y-m-01 00:00:00', strtotime('+1 month', $start))); return [$start, $end]; case 'year': $start = strtotime(date('Y-01-01 00:00:00', $base)); $end = strtotime(date('Y-01-01 00:00:00', strtotime('+1 year', $start))); return [$start, $end]; default: return [$base, $base + 86400]; } } public function GetVisualizationPopup(): string { // Popup (Fullscreen) soll das gleiche HTML wie das Tile anzeigen $this->RecalculateAndPush(); return $this->GetVisualizationTile(); } private function readDelta(int $varId, int $tStart, int $tEnd, array &$dbg): float { $dbg = [ 'varId' => $varId, 'archiveId' => 0, 'count' => 0, // hier: wie viele AC-Calls erfolgreich Werte lieferten (nicht Anzahl Rohwerte) 'vStart' => null, 'vEnd' => null, 'tsStart' => null, 'tsEnd' => null ]; if ($varId <= 0 || !IPS_VariableExists($varId)) { return 0.0; } $archiveID = IPS_GetInstanceListByModuleID('{43192F0B-135B-4CE7-A0A7-1475603F3060}')[0] ?? 0; $dbg['archiveId'] = $archiveID; if ($archiveID <= 0) { return 0.0; } // Liefert den letzten geloggten Wert <= $toTs. // Wir starten mit einem kleinen Suchfenster (1 Tag) und vergrößern bei Bedarf. $getLastUpTo = function (int $toTs) use ($archiveID, $varId): ?array { $windows = [ 86400, // 1 Tag 7 * 86400, // 1 Woche 31 * 86400, // 1 Monat 365 * 86400, // 1 Jahr ]; foreach ($windows as $w) { $from = $toTs - $w; if ($from < 0) $from = 0; // Limit = 1: wir wollen nur einen Wert $vals = @AC_GetLoggedValues($archiveID, $varId, $from, $toTs, 1); if (!empty($vals)) { // In der Praxis liefert Symcon bei Limit=1 den "letzten" im Bereich. // Wir nehmen trotzdem defensiv den Eintrag mit dem größten TimeStamp. $best = $vals[0]; $bestTs = (int)$best['TimeStamp']; foreach ($vals as $v) { $ts = (int)$v['TimeStamp']; if ($ts > $bestTs) { $bestTs = $ts; $best = $v; } } return $best; } } // letzter Versuch: gesamtes Archiv bis toTs (kann bei "total" groß sein, aber passiert nur wenn oben nix gefunden wurde) $vals = @AC_GetLoggedValues($archiveID, $varId, 0, $toTs, 1); if (!empty($vals)) { $best = $vals[0]; $bestTs = (int)$best['TimeStamp']; foreach ($vals as $v) { $ts = (int)$v['TimeStamp']; if ($ts > $bestTs) { $bestTs = $ts; $best = $v; } } return $best; } return null; }; // vStart = letzter Wert <= tStart $a = $getLastUpTo($tStart); if ($a !== null) { $dbg['count']++; $dbg['vStart'] = (float)$a['Value']; $dbg['tsStart'] = (int)$a['TimeStamp']; } else { return 0.0; } // vEnd = letzter Wert <= tEnd $b = $getLastUpTo($tEnd); if ($b !== null) { $dbg['count']++; $dbg['vEnd'] = (float)$b['Value']; $dbg['tsEnd'] = (int)$b['TimeStamp']; } else { return 0.0; } $diff = (float)$dbg['vEnd'] - (float)$dbg['vStart']; return ($diff < 0) ? 0.0 : $diff; } private function getLastLogTimestamp(int $varId): int { if ($varId <= 0 || !IPS_VariableExists($varId)) { return 0; } $archiveID = IPS_GetInstanceListByModuleID('{43192F0B-135B-4CE7-A0A7-1475603F3060}')[0] ?? 0; if ($archiveID <= 0) { return 0; } $values = @AC_GetLoggedValues($archiveID, $varId, 0, time(), 1); if (empty($values)) { return 0; } return (int)$values[0]['TimeStamp']; } private function ShiftDate(string $action): void { $range = $this->ReadAttributeString(self::ATTR_RANGE); if ($range === 'total') { return; } if ($action === 'Today') { $this->WriteAttributeString(self::ATTR_DATE, date('Y-m-d')); return; } $date = $this->ReadAttributeString(self::ATTR_DATE); $base = strtotime($date . ' 00:00:00') ?: strtotime(date('Y-m-d') . ' 00:00:00'); $sign = ($action === 'Prev') ? -1 : 1; switch ($range) { case 'day': $base = strtotime(($sign === -1 ? '-1 day' : '+1 day'), $base); break; case 'week': $base = strtotime(($sign === -1 ? '-7 day' : '+7 day'), $base); break; case 'month': $base = strtotime(($sign === -1 ? '-1 month' : '+1 month'), $base); break; case 'year': $base = strtotime(($sign === -1 ? '-1 year' : '+1 year'), $base); break; } $this->WriteAttributeString(self::ATTR_DATE, date('Y-m-d', $base)); } private function isValidDate(string $ymd): bool { if (!preg_match('/^\d{4}-\d{2}-\d{2}$/', $ymd)) { return false; } [$y, $m, $d] = array_map('intval', explode('-', $ymd)); return checkdate($m, $d, $y); } } ?>