/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright 2013 - 2021, nymea GmbH * Contact: contact@nymea.io * * This fileDescriptor is part of nymea. * This project including source code and documentation is protected by * copyright law, and remains the property of nymea GmbH. All rights, including * reproduction, publication, editing and translation, are reserved. The use of * this project is subject to the terms of a license agreement to be concluded * with nymea GmbH in accordance with the terms of use of nymea GmbH, available * under https://nymea.io/license * * GNU Lesser General Public License Usage * Alternatively, this project may be redistributed and/or modified under the * terms of the GNU Lesser General Public License as published by the Free * Software Foundation; version 3. This project is distributed in the hope that * it will be useful, but WITHOUT ANY WARRANTY; without even the implied * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this project. If not, see . * * For any further details and any questions please contact us under * contact@nymea.io or see our FAQ/Licensing Information on * https://nymea.io/license/faq * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #include "sunspecreactivecurrentmodel.h" #include "sunspecconnection.h" SunSpecReactiveCurrentModel::SunSpecReactiveCurrentModel(SunSpecConnection *connection, quint16 modbusStartRegister, quint16 modelLength, SunSpecDataPoint::ByteOrder byteOrder, QObject *parent) : SunSpecModel(connection, modbusStartRegister, 128, modelLength, byteOrder, parent) { m_modelBlockType = SunSpecModel::ModelBlockTypeFixed; initDataPoints(); } SunSpecReactiveCurrentModel::~SunSpecReactiveCurrentModel() { } QString SunSpecReactiveCurrentModel::name() const { return "reactive_current"; } QString SunSpecReactiveCurrentModel::description() const { return "Dynamic Reactive Current "; } QString SunSpecReactiveCurrentModel::label() const { return "Dynamic Reactive Current"; } SunSpecReactiveCurrentModel::Argramod SunSpecReactiveCurrentModel::arGraMod() const { return m_arGraMod; } QModbusReply *SunSpecReactiveCurrentModel::setArGraMod(Argramod arGraMod) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("ArGraMod"); QVector registers = SunSpecDataPoint::convertFromUInt16(static_cast(arGraMod)); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } float SunSpecReactiveCurrentModel::arGraSag() const { return m_arGraSag; } QModbusReply *SunSpecReactiveCurrentModel::setArGraSag(float arGraSag) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("ArGraSag"); QVector registers = SunSpecDataPoint::convertFromFloatWithSSF(arGraSag, m_arGraSf, dp.dataType()); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } float SunSpecReactiveCurrentModel::arGraSwell() const { return m_arGraSwell; } QModbusReply *SunSpecReactiveCurrentModel::setArGraSwell(float arGraSwell) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("ArGraSwell"); QVector registers = SunSpecDataPoint::convertFromFloatWithSSF(arGraSwell, m_arGraSf, dp.dataType()); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } SunSpecReactiveCurrentModel::ModenaFlags SunSpecReactiveCurrentModel::modEna() const { return m_modEna; } QModbusReply *SunSpecReactiveCurrentModel::setModEna(ModenaFlags modEna) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("ModEna"); QVector registers = SunSpecDataPoint::convertFromUInt16(static_cast(modEna)); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } quint16 SunSpecReactiveCurrentModel::filTms() const { return m_filTms; } QModbusReply *SunSpecReactiveCurrentModel::setFilTms(quint16 filTms) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("FilTms"); QVector registers = SunSpecDataPoint::convertFromUInt16(filTms); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } float SunSpecReactiveCurrentModel::dbVMin() const { return m_dbVMin; } QModbusReply *SunSpecReactiveCurrentModel::setDbVMin(float dbVMin) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("DbVMin"); QVector registers = SunSpecDataPoint::convertFromFloatWithSSF(dbVMin, m_vRefPctSf, dp.dataType()); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } float SunSpecReactiveCurrentModel::dbVMax() const { return m_dbVMax; } QModbusReply *SunSpecReactiveCurrentModel::setDbVMax(float dbVMax) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("DbVMax"); QVector registers = SunSpecDataPoint::convertFromFloatWithSSF(dbVMax, m_vRefPctSf, dp.dataType()); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } float SunSpecReactiveCurrentModel::blkZnV() const { return m_blkZnV; } QModbusReply *SunSpecReactiveCurrentModel::setBlkZnV(float blkZnV) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("BlkZnV"); QVector registers = SunSpecDataPoint::convertFromFloatWithSSF(blkZnV, m_vRefPctSf, dp.dataType()); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } float SunSpecReactiveCurrentModel::hysBlkZnV() const { return m_hysBlkZnV; } QModbusReply *SunSpecReactiveCurrentModel::setHysBlkZnV(float hysBlkZnV) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("HysBlkZnV"); QVector registers = SunSpecDataPoint::convertFromFloatWithSSF(hysBlkZnV, m_vRefPctSf, dp.dataType()); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } quint16 SunSpecReactiveCurrentModel::blkZnTmms() const { return m_blkZnTmms; } QModbusReply *SunSpecReactiveCurrentModel::setBlkZnTmms(quint16 blkZnTmms) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("BlkZnTmms"); QVector registers = SunSpecDataPoint::convertFromUInt16(blkZnTmms); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } quint16 SunSpecReactiveCurrentModel::holdTmms() const { return m_holdTmms; } QModbusReply *SunSpecReactiveCurrentModel::setHoldTmms(quint16 holdTmms) { if (!m_initialized) return nullptr; SunSpecDataPoint dp = m_dataPoints.value("HoldTmms"); QVector registers = SunSpecDataPoint::convertFromUInt16(holdTmms); QModbusDataUnit request = QModbusDataUnit(QModbusDataUnit::RegisterType::HoldingRegisters, m_modbusStartRegister + dp.addressOffset(), registers.length()); request.setValues(registers); return m_connection->modbusTcpClient()->sendWriteRequest(request, m_connection->slaveId()); } qint16 SunSpecReactiveCurrentModel::arGraSf() const { return m_arGraSf; } qint16 SunSpecReactiveCurrentModel::vRefPctSf() const { return m_vRefPctSf; } quint16 SunSpecReactiveCurrentModel::pad() const { return m_pad; } void SunSpecReactiveCurrentModel::initDataPoints() { SunSpecDataPoint modelIdDataPoint; modelIdDataPoint.setName("ID"); modelIdDataPoint.setLabel("Model ID"); modelIdDataPoint.setDescription("Model identifier"); modelIdDataPoint.setMandatory(true); modelIdDataPoint.setSize(1); modelIdDataPoint.setAddressOffset(0); modelIdDataPoint.setSunSpecDataType("uint16"); modelIdDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(modelIdDataPoint.name(), modelIdDataPoint); SunSpecDataPoint modelLengthDataPoint; modelLengthDataPoint.setName("L"); modelLengthDataPoint.setLabel("Model Length"); modelLengthDataPoint.setDescription("Model length"); modelLengthDataPoint.setMandatory(true); modelLengthDataPoint.setSize(1); modelLengthDataPoint.setAddressOffset(1); modelLengthDataPoint.setSunSpecDataType("uint16"); modelLengthDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(modelLengthDataPoint.name(), modelLengthDataPoint); SunSpecDataPoint arGraModDataPoint; arGraModDataPoint.setName("ArGraMod"); arGraModDataPoint.setLabel("ArGraMod"); arGraModDataPoint.setDescription("Indicates if gradients trend toward zero at the edges of the deadband or trend toward zero at the center of the deadband."); arGraModDataPoint.setMandatory(true); arGraModDataPoint.setSize(1); arGraModDataPoint.setAddressOffset(2); arGraModDataPoint.setBlockOffset(0); arGraModDataPoint.setSunSpecDataType("enum16"); arGraModDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); arGraModDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(arGraModDataPoint.name(), arGraModDataPoint); SunSpecDataPoint arGraSagDataPoint; arGraSagDataPoint.setName("ArGraSag"); arGraSagDataPoint.setLabel("ArGraSag"); arGraSagDataPoint.setDescription("The gradient used to increase capacitive dynamic current. A value of 0 indicates no additional reactive current support."); arGraSagDataPoint.setUnits("%ARtg/%dV"); arGraSagDataPoint.setMandatory(true); arGraSagDataPoint.setSize(1); arGraSagDataPoint.setAddressOffset(3); arGraSagDataPoint.setBlockOffset(1); arGraSagDataPoint.setScaleFactorName("ArGra_SF"); arGraSagDataPoint.setSunSpecDataType("uint16"); arGraSagDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); arGraSagDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(arGraSagDataPoint.name(), arGraSagDataPoint); SunSpecDataPoint arGraSwellDataPoint; arGraSwellDataPoint.setName("ArGraSwell"); arGraSwellDataPoint.setLabel("ArGraSwell"); arGraSwellDataPoint.setDescription("The gradient used to increase inductive dynamic current. A value of 0 indicates no additional reactive current support."); arGraSwellDataPoint.setUnits("%ARtg/%dV"); arGraSwellDataPoint.setMandatory(true); arGraSwellDataPoint.setSize(1); arGraSwellDataPoint.setAddressOffset(4); arGraSwellDataPoint.setBlockOffset(2); arGraSwellDataPoint.setScaleFactorName("ArGra_SF"); arGraSwellDataPoint.setSunSpecDataType("uint16"); arGraSwellDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); arGraSwellDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(arGraSwellDataPoint.name(), arGraSwellDataPoint); SunSpecDataPoint modEnaDataPoint; modEnaDataPoint.setName("ModEna"); modEnaDataPoint.setLabel("ModEna"); modEnaDataPoint.setDescription("Activate dynamic reactive current model"); modEnaDataPoint.setMandatory(true); modEnaDataPoint.setSize(1); modEnaDataPoint.setAddressOffset(5); modEnaDataPoint.setBlockOffset(3); modEnaDataPoint.setSunSpecDataType("bitfield16"); modEnaDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); modEnaDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(modEnaDataPoint.name(), modEnaDataPoint); SunSpecDataPoint filTmsDataPoint; filTmsDataPoint.setName("FilTms"); filTmsDataPoint.setLabel("FilTms"); filTmsDataPoint.setDescription("The time window used to calculate the moving average voltage."); filTmsDataPoint.setUnits("Secs"); filTmsDataPoint.setSize(1); filTmsDataPoint.setAddressOffset(6); filTmsDataPoint.setBlockOffset(4); filTmsDataPoint.setSunSpecDataType("uint16"); filTmsDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); filTmsDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(filTmsDataPoint.name(), filTmsDataPoint); SunSpecDataPoint dbVMinDataPoint; dbVMinDataPoint.setName("DbVMin"); dbVMinDataPoint.setLabel("DbVMin"); dbVMinDataPoint.setDescription("The lower delta voltage limit for which negative voltage deviations less than this value no dynamic vars are produced."); dbVMinDataPoint.setUnits("% VRef"); dbVMinDataPoint.setSize(1); dbVMinDataPoint.setAddressOffset(7); dbVMinDataPoint.setBlockOffset(5); dbVMinDataPoint.setScaleFactorName("VRefPct_SF"); dbVMinDataPoint.setSunSpecDataType("uint16"); dbVMinDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); dbVMinDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(dbVMinDataPoint.name(), dbVMinDataPoint); SunSpecDataPoint dbVMaxDataPoint; dbVMaxDataPoint.setName("DbVMax"); dbVMaxDataPoint.setLabel("DbVMax"); dbVMaxDataPoint.setDescription("The upper delta voltage limit for which positive voltage deviations less than this value no dynamic current produced."); dbVMaxDataPoint.setUnits("% VRef"); dbVMaxDataPoint.setSize(1); dbVMaxDataPoint.setAddressOffset(8); dbVMaxDataPoint.setBlockOffset(6); dbVMaxDataPoint.setScaleFactorName("VRefPct_SF"); dbVMaxDataPoint.setSunSpecDataType("uint16"); dbVMaxDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); dbVMaxDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(dbVMaxDataPoint.name(), dbVMaxDataPoint); SunSpecDataPoint blkZnVDataPoint; blkZnVDataPoint.setName("BlkZnV"); blkZnVDataPoint.setLabel("BlkZnV"); blkZnVDataPoint.setDescription("Block zone voltage which defines a lower voltage boundary below which no dynamic current is produced."); blkZnVDataPoint.setUnits("% VRef"); blkZnVDataPoint.setSize(1); blkZnVDataPoint.setAddressOffset(9); blkZnVDataPoint.setBlockOffset(7); blkZnVDataPoint.setScaleFactorName("VRefPct_SF"); blkZnVDataPoint.setSunSpecDataType("uint16"); blkZnVDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); blkZnVDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(blkZnVDataPoint.name(), blkZnVDataPoint); SunSpecDataPoint hysBlkZnVDataPoint; hysBlkZnVDataPoint.setName("HysBlkZnV"); hysBlkZnVDataPoint.setLabel("HysBlkZnV"); hysBlkZnVDataPoint.setDescription("Hysteresis voltage used with BlkZnV."); hysBlkZnVDataPoint.setUnits("% VRef"); hysBlkZnVDataPoint.setSize(1); hysBlkZnVDataPoint.setAddressOffset(10); hysBlkZnVDataPoint.setBlockOffset(8); hysBlkZnVDataPoint.setScaleFactorName("VRefPct_SF"); hysBlkZnVDataPoint.setSunSpecDataType("uint16"); hysBlkZnVDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); hysBlkZnVDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(hysBlkZnVDataPoint.name(), hysBlkZnVDataPoint); SunSpecDataPoint blkZnTmmsDataPoint; blkZnTmmsDataPoint.setName("BlkZnTmms"); blkZnTmmsDataPoint.setLabel("BlkZnTmms"); blkZnTmmsDataPoint.setDescription("Block zone time the time before which reactive current support remains active regardless of how low the voltage drops."); blkZnTmmsDataPoint.setUnits("mSecs"); blkZnTmmsDataPoint.setSize(1); blkZnTmmsDataPoint.setAddressOffset(11); blkZnTmmsDataPoint.setBlockOffset(9); blkZnTmmsDataPoint.setSunSpecDataType("uint16"); blkZnTmmsDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); blkZnTmmsDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(blkZnTmmsDataPoint.name(), blkZnTmmsDataPoint); SunSpecDataPoint holdTmmsDataPoint; holdTmmsDataPoint.setName("HoldTmms"); holdTmmsDataPoint.setLabel("HoldTmms"); holdTmmsDataPoint.setDescription("Hold time during which reactive current support continues after the average voltage has entered the dead zone."); holdTmmsDataPoint.setUnits("mSecs"); holdTmmsDataPoint.setSize(1); holdTmmsDataPoint.setAddressOffset(12); holdTmmsDataPoint.setBlockOffset(10); holdTmmsDataPoint.setSunSpecDataType("uint16"); holdTmmsDataPoint.setAccess(SunSpecDataPoint::AccessReadWrite); holdTmmsDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(holdTmmsDataPoint.name(), holdTmmsDataPoint); SunSpecDataPoint arGraSfDataPoint; arGraSfDataPoint.setName("ArGra_SF"); arGraSfDataPoint.setLabel("ArGra_SF"); arGraSfDataPoint.setDescription("Scale factor for the gradients."); arGraSfDataPoint.setMandatory(true); arGraSfDataPoint.setSize(1); arGraSfDataPoint.setAddressOffset(13); arGraSfDataPoint.setBlockOffset(11); arGraSfDataPoint.setSunSpecDataType("sunssf"); arGraSfDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(arGraSfDataPoint.name(), arGraSfDataPoint); SunSpecDataPoint vRefPctSfDataPoint; vRefPctSfDataPoint.setName("VRefPct_SF"); vRefPctSfDataPoint.setLabel("VRefPct_SF"); vRefPctSfDataPoint.setDescription("Scale factor for the voltage zone and limit settings."); vRefPctSfDataPoint.setSize(1); vRefPctSfDataPoint.setAddressOffset(14); vRefPctSfDataPoint.setBlockOffset(12); vRefPctSfDataPoint.setSunSpecDataType("sunssf"); vRefPctSfDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(vRefPctSfDataPoint.name(), vRefPctSfDataPoint); SunSpecDataPoint padDataPoint; padDataPoint.setName("Pad"); padDataPoint.setSize(1); padDataPoint.setAddressOffset(15); padDataPoint.setBlockOffset(13); padDataPoint.setSunSpecDataType("pad"); padDataPoint.setByteOrder(m_byteOrder); m_dataPoints.insert(padDataPoint.name(), padDataPoint); } void SunSpecReactiveCurrentModel::processBlockData() { // Scale factors if (m_dataPoints.value("ArGra_SF").isValid()) m_arGraSf = m_dataPoints.value("ArGra_SF").toInt16(); if (m_dataPoints.value("VRefPct_SF").isValid()) m_vRefPctSf = m_dataPoints.value("VRefPct_SF").toInt16(); // Update properties according to the data point type if (m_dataPoints.value("ArGraMod").isValid()) m_arGraMod = static_cast(m_dataPoints.value("ArGraMod").toUInt16()); if (m_dataPoints.value("ArGraSag").isValid()) m_arGraSag = m_dataPoints.value("ArGraSag").toFloatWithSSF(m_arGraSf); if (m_dataPoints.value("ArGraSwell").isValid()) m_arGraSwell = m_dataPoints.value("ArGraSwell").toFloatWithSSF(m_arGraSf); if (m_dataPoints.value("ModEna").isValid()) m_modEna = static_cast(m_dataPoints.value("ModEna").toUInt16()); if (m_dataPoints.value("FilTms").isValid()) m_filTms = m_dataPoints.value("FilTms").toUInt16(); if (m_dataPoints.value("DbVMin").isValid()) m_dbVMin = m_dataPoints.value("DbVMin").toFloatWithSSF(m_vRefPctSf); if (m_dataPoints.value("DbVMax").isValid()) m_dbVMax = m_dataPoints.value("DbVMax").toFloatWithSSF(m_vRefPctSf); if (m_dataPoints.value("BlkZnV").isValid()) m_blkZnV = m_dataPoints.value("BlkZnV").toFloatWithSSF(m_vRefPctSf); if (m_dataPoints.value("HysBlkZnV").isValid()) m_hysBlkZnV = m_dataPoints.value("HysBlkZnV").toFloatWithSSF(m_vRefPctSf); if (m_dataPoints.value("BlkZnTmms").isValid()) m_blkZnTmms = m_dataPoints.value("BlkZnTmms").toUInt16(); if (m_dataPoints.value("HoldTmms").isValid()) m_holdTmms = m_dataPoints.value("HoldTmms").toUInt16(); if (m_dataPoints.value("ArGra_SF").isValid()) m_arGraSf = m_dataPoints.value("ArGra_SF").toInt16(); if (m_dataPoints.value("VRefPct_SF").isValid()) m_vRefPctSf = m_dataPoints.value("VRefPct_SF").toInt16(); if (m_dataPoints.value("Pad").isValid()) m_pad = m_dataPoints.value("Pad").toUInt16(); qCDebug(dcSunSpecModelData()) << this; } QDebug operator<<(QDebug debug, SunSpecReactiveCurrentModel *model) { debug.nospace().noquote() << "SunSpecReactiveCurrentModel(Model: " << model->modelId() << ", Register: " << model->modbusStartRegister() << ", Length: " << model->modelLength() << ")\n"; debug.nospace().noquote() << " - " << model->dataPoints().value("ArGraMod") << "-->"; if (model->dataPoints().value("ArGraMod").isValid()) { debug.nospace().noquote() << model->arGraMod() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("ArGraSag") << "-->"; if (model->dataPoints().value("ArGraSag").isValid()) { debug.nospace().noquote() << model->arGraSag() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("ArGraSwell") << "-->"; if (model->dataPoints().value("ArGraSwell").isValid()) { debug.nospace().noquote() << model->arGraSwell() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("ModEna") << "-->"; if (model->dataPoints().value("ModEna").isValid()) { debug.nospace().noquote() << model->modEna() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("FilTms") << "-->"; if (model->dataPoints().value("FilTms").isValid()) { debug.nospace().noquote() << model->filTms() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("DbVMin") << "-->"; if (model->dataPoints().value("DbVMin").isValid()) { debug.nospace().noquote() << model->dbVMin() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("DbVMax") << "-->"; if (model->dataPoints().value("DbVMax").isValid()) { debug.nospace().noquote() << model->dbVMax() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("BlkZnV") << "-->"; if (model->dataPoints().value("BlkZnV").isValid()) { debug.nospace().noquote() << model->blkZnV() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("HysBlkZnV") << "-->"; if (model->dataPoints().value("HysBlkZnV").isValid()) { debug.nospace().noquote() << model->hysBlkZnV() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("BlkZnTmms") << "-->"; if (model->dataPoints().value("BlkZnTmms").isValid()) { debug.nospace().noquote() << model->blkZnTmms() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("HoldTmms") << "-->"; if (model->dataPoints().value("HoldTmms").isValid()) { debug.nospace().noquote() << model->holdTmms() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } debug.nospace().noquote() << " - " << model->dataPoints().value("Pad") << "-->"; if (model->dataPoints().value("Pad").isValid()) { debug.nospace().noquote() << model->pad() << "\n"; } else { debug.nospace().noquote() << "NaN\n"; } return debug.space().quote(); }