nymea-plugins/flowercare/flowercare.cpp

195 lines
7.9 KiB
C++

// SPDX-License-Identifier: GPL-3.0-or-later
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
* Copyright (C) 2013 - 2024, nymea GmbH
* Copyright (C) 2024 - 2025, chargebyte austria GmbH
*
* This file is part of nymea-plugins.
*
* nymea-plugins is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* nymea-plugins 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
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with nymea-plugins. If not, see <https://www.gnu.org/licenses/>.
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
#include "flowercare.h"
#include "extern-plugininfo.h"
#include <QDataStream>
FlowerCare::FlowerCare(BluetoothLowEnergyDevice *thing, QObject *parent):
QObject(parent),
m_bluetoothDevice(thing)
{
connect(m_bluetoothDevice, &BluetoothLowEnergyDevice::connectedChanged, this, &FlowerCare::onConnectedChanged);
connect(m_bluetoothDevice, &BluetoothLowEnergyDevice::servicesDiscoveryFinished, this, &FlowerCare::onServiceDiscoveryFinished);
}
void FlowerCare::refreshData()
{
qCDebug(dcFlowerCare()) << "Connecting to device";
m_bluetoothDevice->connectDevice();
}
BluetoothLowEnergyDevice *FlowerCare::btDevice() const
{
return m_bluetoothDevice;
}
void FlowerCare::onConnectedChanged(bool connected)
{
qCDebug(dcFlowerCare()) << "Connection changed:" << connected;
if (!connected) {
m_sensorService->deleteLater();
m_sensorService = nullptr;
}
}
void FlowerCare::onServiceDiscoveryFinished()
{
BluetoothLowEnergyDevice *btDev = static_cast<BluetoothLowEnergyDevice*>(sender());
qCDebug(dcFlowerCare()) << "have service uuids" << btDev->serviceUuids();
m_sensorService = btDev->controller()->createServiceObject(sensorServiceUuid, this);
connect(m_sensorService, &QLowEnergyService::stateChanged, this, &FlowerCare::onSensorServiceStateChanged);
connect(m_sensorService, &QLowEnergyService::characteristicRead, this, &FlowerCare::onSensorServiceCharacteristicRead);
connect(m_sensorService, &QLowEnergyService::characteristicChanged, this, &FlowerCare::onSensorServiceCharacteristicChanged);
m_sensorService->discoverDetails();
}
void FlowerCare::onSensorServiceStateChanged(const QLowEnergyService::ServiceState &state)
{
if (state != QLowEnergyService::ServiceDiscovered) {
return;
}
// printServiceDetails(m_sensorService);
QLowEnergyCharacteristic batteryFirmwareCharacteristic = m_sensorService->characteristic(batteryFirmwareCharacteristicUuid);
if (!batteryFirmwareCharacteristic.isValid()) {
qCWarning(dcFlowerCare()) << "Invalid battery/firmware characteristic.";
emit failed();
return;
}
QByteArray value = batteryFirmwareCharacteristic.value();
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
QDataStream stream(&value, QDataStream::ReadOnly);
#else
QDataStream stream(&value, QIODevice::ReadOnly);
#endif
stream.setByteOrder(QDataStream::LittleEndian);
stream >> m_batteryLevel;
QString firmwareVersionString = value.right(5);
qCDebug(dcFlowerCare()) << "Battery level:" << m_batteryLevel;
qCDebug(dcFlowerCare()) << "Firmware version:" << firmwareVersionString;
if (firmwareVersionString >= "2.6.6") {
QLowEnergyCharacteristic sensorControlCharacteristic = m_sensorService->characteristic(sensorControlCharacteristicUuid);
m_sensorService->writeCharacteristic(sensorControlCharacteristic, QByteArray::fromHex("A01F"));
qCDebug(dcFlowerCare()) << "Wrote to handle 0x0033: A01F";
}
m_sensorDataCharacteristic = m_sensorService->characteristic(sensorDataCharacteristicUuid);
if (!m_sensorDataCharacteristic.isValid()) {
qCWarning(dcFlowerCare()) << "Invalid sensor data characteristic.";
}
// Enable notifications
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
QLowEnergyDescriptor notificationDescriptor = m_sensorDataCharacteristic.descriptor(QBluetoothUuid::DescriptorType::ClientCharacteristicConfiguration);
#else
QLowEnergyDescriptor notificationDescriptor = m_sensorDataCharacteristic.descriptor(QBluetoothUuid::ClientCharacteristicConfiguration);
#endif
m_sensorService->writeDescriptor(notificationDescriptor, QByteArray::fromHex("0100"));
// Read the data manually
// Sometimes if we read the light intensty right now we might get wrong values
// because the LED flashes upon connect. Let's not read it manually but instead wait for
// the values to come in with the notification
// m_sensorService->readCharacteristic(m_sensorDataCharacteristic);
}
void FlowerCare::onSensorServiceCharacteristicRead(const QLowEnergyCharacteristic &characteristic, const QByteArray &value)
{
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
qCDebug(dcFlowerCare()) << "Characteristic read" << characteristic.uuid().toString() << value.toHex();
#else
qCDebug(dcFlowerCare()) << "Characteristic read" << QString::number(characteristic.handle(), 16) << value.toHex();
#endif
if (characteristic != m_sensorDataCharacteristic) {
return;
}
processSensorData(value);
}
void FlowerCare::onSensorServiceCharacteristicChanged(const QLowEnergyCharacteristic &characteristic, const QByteArray &value)
{
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
qCDebug(dcFlowerCare()) << "Notification received" << characteristic.uuid().toString() << value.toHex();
#else
qCDebug(dcFlowerCare()) << "Notification received" << QString::number(characteristic.handle(), 16) << value.toHex();
#endif
if (characteristic != m_sensorDataCharacteristic) {
return;
}
processSensorData(value);
}
void FlowerCare::printServiceDetails(QLowEnergyService *service) const
{
foreach (const QLowEnergyCharacteristic &characteristic, service->characteristics()) {
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
qCDebug(dcFlowerCare()).nospace() << "C: --> " << characteristic.uuid().toString() << " (" << " Name: " << characteristic.name() << "): " << characteristic.value() << ", " << characteristic.value().toHex();
#else
qCDebug(dcFlowerCare()).nospace() << "C: --> " << characteristic.uuid().toString() << " (" << characteristic.handle() << " Name: " << characteristic.name() << "): " << characteristic.value() << ", " << characteristic.value().toHex();
#endif
foreach (const QLowEnergyDescriptor &descriptor, characteristic.descriptors()) {
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
qCDebug(dcFlowerCare()).nospace() << "D: --> " << descriptor.uuid().toString() << " (" << " Name: " << descriptor.name() << "): " << descriptor.value() << ", " << descriptor.value().toHex();
#else
qCDebug(dcFlowerCare()).nospace() << "D: --> " << descriptor.uuid().toString() << " (" << descriptor.handle() << " Name: " << descriptor.name() << "): " << descriptor.value() << ", " << descriptor.value().toHex();
#endif
}
}
}
void FlowerCare::processSensorData(const QByteArray &data)
{
QByteArray copy = data;
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
QDataStream stream(&copy, QDataStream::ReadOnly);
#else
QDataStream stream(&copy, QIODevice::ReadOnly);
#endif
stream.setByteOrder(QDataStream::LittleEndian);
qint16 temp;
stream >> temp;
qint8 skip;
stream >> skip;
quint32 lux;
stream >> lux;
qint8 moisture;
stream >> moisture;
qint16 fertility;
stream >> fertility;
qCDebug(dcFlowerCare()) << "Temperature:" << temp << "Lux:" << lux << "moisture:" << moisture << "fertility" << fertility;
m_bluetoothDevice->disconnectDevice();
emit finished(m_batteryLevel, 1.0 * temp / 10, lux, moisture, fertility);
}