161 lines
5.4 KiB
PHP
161 lines
5.4 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
class PV_Visu extends IPSModule
|
|
{
|
|
public function Create()
|
|
{
|
|
parent::Create();
|
|
// Vier Zähler-Variablen
|
|
$this->RegisterPropertyInteger('VarProduction', 0);
|
|
$this->RegisterPropertyInteger('VarConsumption', 0);
|
|
$this->RegisterPropertyInteger('VarFeedIn', 0);
|
|
$this->RegisterPropertyInteger('VarGrid', 0);
|
|
// HTML-SDK Tile aktivieren
|
|
$this->SetVisualizationType(1);
|
|
}
|
|
|
|
public function ApplyChanges(): void
|
|
{
|
|
parent::ApplyChanges();
|
|
// Auf Änderungen der Zähler-Variablen reagieren
|
|
foreach (['VarProduction', 'VarConsumption', 'VarFeedIn', 'VarGrid'] as $prop) {
|
|
$vid = $this->ReadPropertyInteger($prop);
|
|
if ($vid > 0) {
|
|
// Register event for variable update
|
|
$this->RegisterMessage(\$vid, VM_UPDATE);
|
|
}
|
|
}
|
|
}
|
|
|
|
public function MessageSink(int $TimeStamp, int $SenderID, int $Message, $Data): void
|
|
{
|
|
if ($Message === VM_UPDATE) {
|
|
// bei jeder Aktualisierung einer Zählervariable neu senden
|
|
$this->UpdateData();
|
|
}
|
|
}(int $TimeStamp, int $SenderID, int $Message, $Data): void
|
|
{
|
|
if ($Message === VM_UPDATE) {
|
|
// bei jeder Aktualisierung einer Zählervariable neu senden
|
|
$this->UpdateData();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Liefert das HTML-Template und injiziert initiale Daten
|
|
*/
|
|
public function GetVisualizationTile(): string
|
|
{
|
|
$start = strtotime('today 00:00');
|
|
$end = time();
|
|
|
|
$prod = $this->GetDailyTotal($this->ReadPropertyInteger('VarProduction'), $start, $end);
|
|
$cons = $this->GetDailyTotal($this->ReadPropertyInteger('VarConsumption'), $start, $end);
|
|
$feed = $this->GetDailyTotal($this->ReadPropertyInteger('VarFeedIn'), $start, $end);
|
|
$grid = $this->GetDailyTotal($this->ReadPropertyInteger('VarGrid'), $start, $end);
|
|
|
|
// Prozent-Quoten
|
|
$prodCons = $prod > 0 ? ($cons / $prod) * 100 : 0;
|
|
$prodFeed = $prod > 0 ? ($feed / $prod) * 100 : 0;
|
|
$consPV = $cons > 0 ? min($prod, $cons) / $cons * 100 : 0;
|
|
$consGrid = $cons > 0 ? ($grid / $cons) * 100 : 0;
|
|
|
|
$data = [
|
|
'prodCons' => round($prodCons, 1),
|
|
'prodFeed' => round($prodFeed, 1),
|
|
'consPV' => round($consPV, 1),
|
|
'consGrid' => round($consGrid, 1),
|
|
'value' => [
|
|
'prod' => round($prod, 2),
|
|
'cons' => round($cons, 2),
|
|
'feed' => round($feed, 2),
|
|
'grid' => round($grid, 2),
|
|
],
|
|
];
|
|
|
|
// Daten als JSON-String übergeben
|
|
$json = json_encode($data);
|
|
IPS_LogMessage("TEST", print_r($data));
|
|
|
|
$this->UpdateVisualizationValue($json);
|
|
|
|
// HTML-Template laden und zurückgeben
|
|
$htmlPath = __DIR__ . '/module.html';
|
|
if (!file_exists($htmlPath)) {
|
|
$this->LogMessage("module.html nicht gefunden in $htmlPath", KL_ERROR);
|
|
return '';
|
|
}
|
|
return file_get_contents($htmlPath);
|
|
}
|
|
|
|
/**
|
|
* Callback vom HTML: sendet frische Daten
|
|
*/
|
|
public function RequestAction($Ident, $Value): void
|
|
{
|
|
if ($Ident === 'update') {
|
|
$this->UpdateData();
|
|
} else {
|
|
throw new \UnexpectedValueException("Unknown Ident $Ident");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Aktualisiert Daten und sendet an Tile
|
|
*/
|
|
public function UpdateData(): void
|
|
{
|
|
$start = strtotime('today 00:00');
|
|
$end = time();
|
|
|
|
$prod = $this->GetDailyTotal($this->ReadPropertyInteger('VarProduction'), $start, $end);
|
|
$cons = $this->GetDailyTotal($this->ReadPropertyInteger('VarConsumption'), $start, $end);
|
|
$feed = $this->GetDailyTotal($this->ReadPropertyInteger('VarFeedIn'), $start, $end);
|
|
$grid = $this->GetDailyTotal($this->ReadPropertyInteger('VarGrid'), $start, $end);
|
|
|
|
$prodCons = $prod > 0 ? ($cons / $prod) * 100 : 0;
|
|
$prodFeed = $prod > 0 ? ($feed / $prod) * 100 : 0;
|
|
$consPV = $cons > 0 ? min($prod, $cons) / $cons * 100 : 0;
|
|
$consGrid = $cons > 0 ? ($grid / $cons) * 100 : 0;
|
|
|
|
$data = [
|
|
'prodCons' => round($prodCons, 1),
|
|
'prodFeed' => round($prodFeed, 1),
|
|
'consPV' => round($consPV, 1),
|
|
'consGrid' => round($consGrid, 1),
|
|
'value' => [
|
|
'prod' => round($prod, 2),
|
|
'cons' => round($cons, 2),
|
|
'feed' => round($feed, 2),
|
|
'grid' => round($grid, 2),
|
|
],
|
|
];
|
|
|
|
$json = json_encode($data);
|
|
IPS_LogMessage("TEST", print_r($data));
|
|
$this->UpdateVisualizationValue($json);
|
|
}
|
|
|
|
/**
|
|
* Holt Tages-Aggregat aus dem IPS-Archiv
|
|
*/
|
|
private function GetDailyTotal(int $varID, int $start, int $end): float
|
|
{
|
|
if ($varID <= 0) {
|
|
return 0.0;
|
|
}
|
|
$archives = IPS_GetInstanceListByModuleID('{43192F0B-135B-4CE7-A0A7-1475603F3060}');
|
|
if (empty($archives)) {
|
|
return 0.0;
|
|
}
|
|
$values = AC_GetAggregatedValues($archives[0], $varID, 1, $start, $end, 1);
|
|
if (empty($values)) {
|
|
return 0.0;
|
|
}
|
|
// Für Zähler den Summenwert nutzen
|
|
return (float)$values[0]['Avg'];
|
|
}
|
|
} |