diff --git a/Energy_Pie/module.html b/Energy_Pie/module.html index dc7f4bd..57e7871 100644 --- a/Energy_Pie/module.html +++ b/Energy_Pie/module.html @@ -1,4 +1,6 @@ -
+ +
- -
-
-
+ style="margin-top:14px;display:grid;grid-template-columns:repeat(2,minmax(0,1fr)); + gap:14px;max-width:880px;">
-
-
- - - - diff --git a/Energy_Pie/module.php b/Energy_Pie/module.php index 9dd6c01..e595682 100644 --- a/Energy_Pie/module.php +++ b/Energy_Pie/module.php @@ -1,57 +1,44 @@ + 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', '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'; @@ -60,7 +47,6 @@ class Energy_Pie extends IPSModule } return file_get_contents($path); } - public function RequestAction($Ident, $Value): void { switch ($Ident) { @@ -72,7 +58,6 @@ class Energy_Pie extends IPSModule $this->WriteAttributeString(self::ATTR_RANGE, $range); $this->RecalculateAndPush(); break; - case 'SetDate': $date = (string)$Value; if (!$this->isValidDate($date)) { @@ -81,62 +66,48 @@ class Energy_Pie extends IPSModule $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); - // Auto-jump to last log day if none found if (!$hasData && $range !== 'total') { $lastTs = $this->getLastLogTimestamp($this->ReadPropertyInteger('VarProduction')); if ($lastTs > 0) { $this->WriteAttributeString(self::ATTR_DATE, date('Y-m-d', $lastTs)); $date = $this->ReadAttributeString(self::ATTR_DATE); - [$tStart, $tEnd] = $this->getRange($range, $date); - $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); } } - // House = Prod - Feed + Grid $house = $prod - $feed + $grid; if ($house < 0) $house = 0.0; - $payload = [ 'range' => $range, 'date' => $date, @@ -156,46 +127,36 @@ class Energy_Pie extends IPSModule '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]; - 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 = [ @@ -207,84 +168,65 @@ class Energy_Pie extends IPSModule 'vStart' => null, 'vEnd' => 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; } - $values = @AC_GetLoggedValues($archiveID, $varId, $tStart - 86400, $tEnd + 86400, 0); if (empty($values)) { return 0.0; } - usort($values, static fn($a, $b) => (int)$a['TimeStamp'] <=> (int)$b['TimeStamp']); $dbg['count'] = count($values); - $dbg['first'] = (float)$values[0]['Value']; $dbg['last'] = (float)$values[count($values) - 1]['Value']; - $vStart = null; $vEnd = null; - foreach ($values as $v) { $ts = (int)$v['TimeStamp']; if ($ts <= $tStart) $vStart = (float)$v['Value']; if ($ts <= $tEnd) $vEnd = (float)$v['Value']; if ($ts > $tEnd) break; } - if ($vStart === null) $vStart = $dbg['first']; if ($vEnd === null) $vEnd = $dbg['last']; - $dbg['vStart'] = $vStart; $dbg['vEnd'] = $vEnd; - $diff = $vEnd - $vStart; return ($diff < 0) ? 0.0 : (float)$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); @@ -296,10 +238,8 @@ class Energy_Pie extends IPSModule $base = strtotime(($sign === -1 ? '-1 month' : '+1 month'), $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)) { @@ -311,5 +251,4 @@ class Energy_Pie extends IPSModule } - -?> \ No newline at end of file +?>