diff --git a/Abrechnung/module.php b/Abrechnung/module.php index d7a7419..6f8849a 100644 --- a/Abrechnung/module.php +++ b/Abrechnung/module.php @@ -284,19 +284,20 @@ private function AddMeterToPDFRow($meter, $tariffs, $from, $to, $type) { $rows = ''; $totalCost = 0.0; - $varId = $meter['var_consumption']; - IPS_LogMessage('Abrechnung', "đ§ [AddMeterToPDFRow] Starte Berechnung fĂźr {$meter['name']} ($type) von " . date('d.m.Y H:i', $from) . " bis " . date('d.m.Y H:i', $to)); + $varId = $meter['var_consumption']; + IPS_LogMessage('Abrechnung', "đ§ [AddMeterToPDFRow] Start fĂźr Zähler '{$meter['name']}' (Typ: {$type}), Zeitraum: " . date('d.m.Y H:i', $from) . " - " . date('d.m.Y H:i', $to) . ", VarID={$varId}"); if (!IPS_VariableExists($varId)) { - IPS_LogMessage('Abrechnung', "â Variable {$varId} fĂźr {$meter['name']} nicht gefunden"); + IPS_LogMessage('Abrechnung', "â [AddMeterToPDFRow] Variable {$varId} fĂźr {$meter['name']} nicht gefunden"); return ['row' => '', 'value' => 0]; } - // --- 1. Relevante Tarife filtern --- + // --- Relevante Tarife nach Typ filtern (LOGGING ONLY) --- $filteredTariffs = array_filter($tariffs, function ($t) use ($type) { - $typeNorm = strtolower(trim($type)); $tariffType = strtolower(trim($t['unit_type'] ?? '')); + $typeNorm = strtolower(trim($type)); + $map = [ 'strombezug' => ['strom', 'strombezug', 'stromzähler', 'stromverbrauch'], 'warmwasser' => ['warmwasser', 'wasser', 'hww'], @@ -306,50 +307,47 @@ private function AddMeterToPDFRow($meter, $tariffs, $from, $to, $type) return in_array($tariffType, $map[$typeNorm] ?? []) || $tariffType === $typeNorm; }); - IPS_LogMessage('Abrechnung', "đš Gefundene Tarife fĂźr Typ '$type': " . count($filteredTariffs)); + IPS_LogMessage('Abrechnung', "đš [AddMeterToPDFRow] passende Tarife fĂźr '{$type}': " . count($filteredTariffs)); if (empty($filteredTariffs)) { - IPS_LogMessage('Abrechnung', "â Keine passenden Tarife gefunden â alles 0 Rp"); + IPS_LogMessage('Abrechnung', "â [AddMeterToPDFRow] Keine passenden Tarife fĂźr Typ '{$type}' gefunden â 0 Rp"); return ['row' => '', 'value' => 0]; } - // --- 2. Zeitstempel normalisieren & sortieren --- + // --- Tarife sicher in Zeiträume umwandeln (LOGGING ONLY) --- foreach ($filteredTariffs as &$t) { $t['start_ts'] = is_numeric($t['start']) ? intval($t['start']) : strtotime($t['start']); $t['end_ts'] = is_numeric($t['end']) ? intval($t['end']) : strtotime($t['end']); IPS_LogMessage('Abrechnung', sprintf( - " đ Tarif: %s - %s @ %.3f Rp", - date('d.m.Y H:i', $t['start_ts']), - date('d.m.Y H:i', $t['end_ts']), - floatval($t['price']) + " đ Tariffenster: %s â %s | Preis: %s Rp | unit_type='%s'", + $t['start_ts'] ? date('d.m.Y H:i', $t['start_ts']) : 'â', + $t['end_ts'] ? date('d.m.Y H:i', $t['end_ts']) : 'â', + isset($t['price']) ? str_replace('.', ',', (string)$t['price']) : 'â', + $t['unit_type'] ?? 'â' )); } unset($t); + + // Sortiere Tarife nach Startzeit (unverändert) usort($filteredTariffs, fn($a, $b) => $a['start_ts'] <=> $b['start_ts']); - // --- 3. Iteration Ăźber Tarifabschnitte --- $current = $from; $segIndex = 1; while ($current < $to) { - IPS_LogMessage('Abrechnung', "âĄď¸ [Segment $segIndex] Start: " . date('d.m.Y H:i', $current)); + IPS_LogMessage('Abrechnung', "âĄď¸ [AddMeterToPDFRow] Segment {$segIndex} START: " . date('d.m.Y H:i', $current)); - // Aktiven Tarif finden - $active = null; + // Finde aktiven Tarif fĂźr diesen Zeitpunkt (unverändert) + $activeTariff = null; foreach ($filteredTariffs as $t) { - if ($t['start_ts'] <= $current && $current < $t['end_ts']) { - $active = $t; + if ($t['start_ts'] <= $current && $current <= $t['end_ts']) { + $activeTariff = $t; break; } } - if ($active) { - $segmentEnd = min($active['end_ts'], $to); - $tariffPrice = floatval($active['price']); - $tariffLabel = number_format($tariffPrice, 2); - IPS_LogMessage('Abrechnung', sprintf(" â Aktiver Tarif gefunden: %.3f Rp (gĂźltig bis %s)", $tariffPrice, date('d.m.Y H:i', $active['end_ts']))); - } else { - // Kein aktiver Tarif gefunden â Nulltarif + // Wenn kein aktiver Tarif, nimm nächsten oder rechne 0 Rp (unverändert) + if (!$activeTariff) { $nextStart = $to; foreach ($filteredTariffs as $t) { if ($t['start_ts'] > $current && $t['start_ts'] < $nextStart) { @@ -359,33 +357,49 @@ private function AddMeterToPDFRow($meter, $tariffs, $from, $to, $type) $segmentEnd = min($nextStart, $to); $tariffPrice = 0.0; $tariffLabel = 'kein Tarif'; - IPS_LogMessage('Abrechnung', sprintf(" â Kein Tarif aktiv â verwende 0 Rp bis %s", date('d.m.Y H:i', $segmentEnd))); + IPS_LogMessage('Abrechnung', " â Kein aktiver Tarif â 0 Rp bis " . date('d.m.Y H:i', $segmentEnd)); + } else { + $segmentEnd = min($activeTariff['end_ts'], $to); + $tariffPrice = floatval($activeTariff['price']); + $tariffLabel = number_format($tariffPrice, 2, ',', ''); + IPS_LogMessage('Abrechnung', sprintf( + " â Aktiver Tarif: %.3f Rp | gĂźltig bis %s | SegmentEnd=%s", + $tariffPrice, + date('d.m.Y H:i', $activeTariff['end_ts']), + date('d.m.Y H:i', $segmentEnd) + )); } + // Sicherheitsnetz (unverändert) if ($segmentEnd <= $current) { - IPS_LogMessage('Abrechnung', " â SegmentEnd <= current â Abbruch (Ende erreicht oder TariflĂźcke fehlerhaft)"); + IPS_LogMessage('Abrechnung', " â segmentEnd <= current (" . date('d.m.Y H:i', $segmentEnd) . " <= " . date('d.m.Y H:i', $current) . "), Abbruch der Schleife."); break; } - // --- 4. Werte holen --- - $startVal = $this->GetValueAt($varId, $current, false); + // --- Verbrauch fĂźr diesen Zeitraum (unverändert) --- + IPS_LogMessage('Abrechnung', " đ Hole Startwert @ " . date('d.m.Y H:i', $current) . " (nearestAfter=true)"); + $startVal = $this->GetValueAt($varId, $current, true); + + IPS_LogMessage('Abrechnung', " đ Hole Endwert @ " . date('d.m.Y H:i', $segmentEnd) . " (nearestAfter=true)"); $endVal = $this->GetValueAt($varId, $segmentEnd, true); + + if ($startVal === null || $endVal === null) { + IPS_LogMessage('Abrechnung', " â Keine Werte gefunden (startVal=" . var_export($startVal, true) . ", endVal=" . var_export($endVal, true) . ") â Segment Ăźbersprungen"); + $current = $segmentEnd + 1; + $segIndex++; + continue; + } + $diff = max(0, $endVal - $startVal); $costCHF = round(($tariffPrice / 100) * $diff, 2); $totalCost += $costCHF; IPS_LogMessage('Abrechnung', sprintf( - " đ Werte: Start=%.3f (%.s), Ende=%.3f (%.s), Verbrauch=%.3f, Tarif=%.3f Rp, Kosten=%.3f CHF", - $startVal, - date('d.m.Y H:i', $current), - $endVal, - date('d.m.Y H:i', $segmentEnd), - $diff, - $tariffPrice, - $costCHF + " đ Segment %d: Start=%.3f, Ende=%.3f, Verbrauch=%.3f, Tarif=%s, Kosten=%.2f CHF", + $segIndex, $startVal, $endVal, $diff, $tariffLabel, $costCHF )); - // --- 5. HTML-Zeile erstellen --- + // --- Ausgabezeile (unverändert) --- $rows .= "