From 7928f2f8cfd8a71cbc72cd316c29de0938ccc14c Mon Sep 17 00:00:00 2001 From: Bernhard Kirchen Date: Sat, 30 Dec 2023 16:46:44 +0100 Subject: [PATCH] Feature: JK BMS: add more values to live view (#552) there are more interesting values available to display in the live view. however, adding them made the list of values very long. this can be mitigated by using a new column/card, which uses the available screen space nicely on bigger screens. --- include/BatteryStats.h | 10 --- src/BatteryStats.cpp | 94 +++++++++++++++------------ webapp/src/components/BatteryView.vue | 6 +- webapp/src/locales/de.json | 10 ++- webapp/src/locales/en.json | 10 ++- webapp/src/locales/fr.json | 10 ++- webapp/src/types/BatteryDataStatus.ts | 4 +- 7 files changed, 81 insertions(+), 63 deletions(-) diff --git a/include/BatteryStats.h b/include/BatteryStats.h index 1ba3b7f79..5d7067208 100644 --- a/include/BatteryStats.h +++ b/include/BatteryStats.h @@ -28,16 +28,6 @@ class BatteryStats { bool isValid() const { return _lastUpdateSoC > 0 && _lastUpdate > 0; } protected: - template - void addLiveViewValue(JsonVariant& root, std::string const& name, - T&& value, std::string const& unit, uint8_t precision) const; - void addLiveViewText(JsonVariant& root, std::string const& name, - std::string const& text) const; - void addLiveViewWarning(JsonVariant& root, std::string const& name, - bool warning) const; - void addLiveViewAlarm(JsonVariant& root, std::string const& name, - bool alarm) const; - String _manufacturer = "unknown"; uint8_t _SoC = 0; uint32_t _lastUpdateSoC = 0; diff --git a/src/BatteryStats.cpp b/src/BatteryStats.cpp index 34e2589ba..9e975a9c6 100644 --- a/src/BatteryStats.cpp +++ b/src/BatteryStats.cpp @@ -7,30 +7,44 @@ #include "JkBmsDataPoints.h" template -void BatteryStats::addLiveViewValue(JsonVariant& root, std::string const& name, - T&& value, std::string const& unit, uint8_t precision) const +static void addLiveViewInSection(JsonVariant& root, + std::string const& section, std::string const& name, + T&& value, std::string const& unit, uint8_t precision) { - auto jsonValue = root["values"][name]; + auto jsonValue = root["values"][section][name]; jsonValue["v"] = value; jsonValue["u"] = unit; jsonValue["d"] = precision; } -void BatteryStats::addLiveViewText(JsonVariant& root, std::string const& name, - std::string const& text) const +template +static void addLiveViewValue(JsonVariant& root, std::string const& name, + T&& value, std::string const& unit, uint8_t precision) +{ + addLiveViewInSection(root, "status", name, value, unit, precision); +} + +static void addLiveViewTextInSection(JsonVariant& root, + std::string const& section, std::string const& name, std::string const& text) { - root["values"][name] = text; + root["values"][section][name] = text; } -void BatteryStats::addLiveViewWarning(JsonVariant& root, std::string const& name, - bool warning) const +static void addLiveViewTextValue(JsonVariant& root, std::string const& name, + std::string const& text) +{ + addLiveViewTextInSection(root, "status", name, text); +} + +static void addLiveViewWarning(JsonVariant& root, std::string const& name, + bool warning) { if (!warning) { return; } root["issues"][name] = 1; } -void BatteryStats::addLiveViewAlarm(JsonVariant& root, std::string const& name, - bool alarm) const +static void addLiveViewAlarm(JsonVariant& root, std::string const& name, + bool alarm) { if (!alarm) { return; } root["issues"][name] = 2; @@ -57,9 +71,9 @@ void PylontechBatteryStats::getLiveViewData(JsonVariant& root) const addLiveViewValue(root, "current", _current, "A", 1); addLiveViewValue(root, "temperature", _temperature, "°C", 1); - addLiveViewText(root, "chargeEnabled", (_chargeEnabled?"yes":"no")); - addLiveViewText(root, "dischargeEnabled", (_dischargeEnabled?"yes":"no")); - addLiveViewText(root, "chargeImmediately", (_chargeImmediately?"yes":"no")); + addLiveViewTextValue(root, "chargeEnabled", (_chargeEnabled?"yes":"no")); + addLiveViewTextValue(root, "dischargeEnabled", (_dischargeEnabled?"yes":"no")); + addLiveViewTextValue(root, "chargeImmediately", (_chargeImmediately?"yes":"no")); // alarms and warnings go into the "Issues" card of the web application addLiveViewWarning(root, "highCurrentDischarge", _warningHighCurrentDischarge); @@ -108,38 +122,11 @@ void JkBmsBatteryStats::getJsonData(JsonVariant& root, bool verbose) const addLiveViewValue(root, "power", current * voltage , "W", 2); } - if (verbose) { - auto oTemperatureOne = _dataPoints.get(); - if (oTemperatureOne.has_value()) { - addLiveViewValue(root, "batOneTemp", *oTemperatureOne, "°C", 0); - } - } - - if (verbose) { - auto oTemperatureTwo = _dataPoints.get(); - if (oTemperatureTwo.has_value()) { - addLiveViewValue(root, "batTwoTemp", *oTemperatureTwo, "°C", 0); - } - } - auto oTemperatureBms = _dataPoints.get(); if (oTemperatureBms.has_value()) { addLiveViewValue(root, "bmsTemp", *oTemperatureBms, "°C", 0); } - if (_cellVoltageTimestamp > 0) { - if (verbose) { - addLiveViewValue(root, "cellMinVoltage", static_cast(_cellMinMilliVolt)/1000, "V", 3); - } - - addLiveViewValue(root, "cellAvgVoltage", static_cast(_cellAvgMilliVolt)/1000, "V", 3); - - if (verbose) { - addLiveViewValue(root, "cellMaxVoltage", static_cast(_cellMaxMilliVolt)/1000, "V", 3); - addLiveViewValue(root, "cellDiffVoltage", (_cellMaxMilliVolt - _cellMinMilliVolt), "mV", 0); - } - } - // labels BatteryChargeEnabled, BatteryDischargeEnabled, and // BalancingEnabled refer to the user setting. we want to show the // actual MOSFETs' state which control whether charging and discharging @@ -148,11 +135,32 @@ void JkBmsBatteryStats::getJsonData(JsonVariant& root, bool verbose) const if (oStatus.has_value()) { using Bits = JkBms::StatusBits; auto chargeEnabled = *oStatus & static_cast(Bits::ChargingActive); - addLiveViewText(root, "chargeEnabled", (chargeEnabled?"yes":"no")); + addLiveViewTextValue(root, "chargeEnabled", (chargeEnabled?"yes":"no")); auto dischargeEnabled = *oStatus & static_cast(Bits::DischargingActive); - addLiveViewText(root, "dischargeEnabled", (dischargeEnabled?"yes":"no")); + addLiveViewTextValue(root, "dischargeEnabled", (dischargeEnabled?"yes":"no")); + } + + auto oTemperatureOne = _dataPoints.get(); + if (oTemperatureOne.has_value()) { + addLiveViewInSection(root, "cells", "batOneTemp", *oTemperatureOne, "°C", 0); + } + + auto oTemperatureTwo = _dataPoints.get(); + if (oTemperatureTwo.has_value()) { + addLiveViewInSection(root, "cells", "batTwoTemp", *oTemperatureTwo, "°C", 0); + } + + if (_cellVoltageTimestamp > 0) { + addLiveViewInSection(root, "cells", "cellMinVoltage", static_cast(_cellMinMilliVolt)/1000, "V", 3); + addLiveViewInSection(root, "cells", "cellAvgVoltage", static_cast(_cellAvgMilliVolt)/1000, "V", 3); + addLiveViewInSection(root, "cells", "cellMaxVoltage", static_cast(_cellMaxMilliVolt)/1000, "V", 3); + addLiveViewInSection(root, "cells", "cellDiffVoltage", (_cellMaxMilliVolt - _cellMinMilliVolt), "mV", 0); + } + + if (oStatus.has_value()) { + using Bits = JkBms::StatusBits; auto balancingActive = *oStatus & static_cast(Bits::BalancingActive); - addLiveViewText(root, "balancingActive", (balancingActive?"yes":"no")); + addLiveViewTextInSection(root, "cells", "balancingActive", (balancingActive?"yes":"no")); } auto oAlarms = _dataPoints.get(); diff --git a/webapp/src/components/BatteryView.vue b/webapp/src/components/BatteryView.vue index 2c3bf11a9..b1fbcfe24 100644 --- a/webapp/src/components/BatteryView.vue +++ b/webapp/src/components/BatteryView.vue @@ -27,9 +27,9 @@
-
+
-
{{ $t('battery.Status') }}
+
{{ $t('battery.' + section) }}
@@ -40,7 +40,7 @@ - +
{{ $t('battery.' + key) }}