/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright 2013 - 2020, nymea GmbH * Contact: contact@nymea.io * * This file 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 "integrationpluginelro.h" #include "plugininfo.h" #include "hardware/radio433/radio433.h" #include #include IntegrationPluginElro::IntegrationPluginElro() { } void IntegrationPluginElro::executeAction(ThingActionInfo *info) { Thing *thing = info->thing(); Action action = info->action(); if (!hardwareManager()->radio433()->available()) return info->finish(Thing::ThingErrorHardwareNotAvailable); if (action.actionTypeId() != elroSocketPowerActionTypeId) return info->finish(Thing::ThingErrorActionTypeNotFound); QList rawData; QByteArray binCode; // create the bincode // channels if (thing->paramValue(elroSocketThingChan1ParamTypeId).toBool()) { binCode.append("00"); } else { binCode.append("01"); } if (thing->paramValue(elroSocketThingChan2ParamTypeId).toBool()) { binCode.append("00"); } else { binCode.append("01"); } if (thing->paramValue(elroSocketThingChan3ParamTypeId).toBool()) { binCode.append("00"); }else{ binCode.append("01"); } if(thing->paramValue(elroSocketThingChan4ParamTypeId).toBool()){ binCode.append("00"); } else { binCode.append("01"); } if (thing->paramValue(elroSocketThingChan5ParamTypeId).toBool()) { binCode.append("00"); } else { binCode.append("01"); } // Buttons if (thing->paramValue(elroSocketThingAParamTypeId).toBool()) { binCode.append("00"); } else { binCode.append("01"); } if (thing->paramValue(elroSocketThingBParamTypeId).toBool()) { binCode.append("00"); } else { binCode.append("01"); } if (thing->paramValue(elroSocketThingCParamTypeId).toBool()) { binCode.append("00"); } else { binCode.append("01"); } if (thing->paramValue(elroSocketThingDParamTypeId).toBool()) { binCode.append("00"); } else { binCode.append("01"); } if (thing->paramValue(elroSocketThingEParamTypeId).toBool()) { binCode.append("00"); } else { binCode.append("01"); } // Power if (action.param(elroSocketPowerActionPowerParamTypeId).value().toBool()) { binCode.append("0001"); } else { binCode.append("0100"); } //create rawData timings list int delay = 350; // sync signal rawData.append(1); rawData.append(31); // add the code foreach (QChar c, binCode) { if (c == '0') { rawData.append(1); rawData.append(3); } else { rawData.append(3); rawData.append(1); } } // send data to hardware resource if (hardwareManager()->radio433()->sendData(delay, rawData, 10)) { qCDebug(dcElro) << "Transmitted" << pluginName() << thing->name() << "power: " << action.param(elroSocketPowerActionPowerParamTypeId).value().toBool(); } else { qCWarning(dcElro) << "Could not transmitt" << pluginName() << thing->name() << "power: " << action.param(elroSocketPowerActionPowerParamTypeId).value().toBool(); return info->finish(Thing::ThingErrorHardwareNotAvailable); } info->finish(Thing::ThingErrorNoError); } void IntegrationPluginElro::radioData(const QList &rawData) { // filter right here a wrong signal length if (rawData.length() != 49) { return; } int delay = rawData.first()/31; QByteArray binCode; // average 314 if (delay > 290 && delay < 400) { // go trough all 48 timings (without sync signal) for (int i = 1; i <= 48; i+=2 ) { int div; int divNext; // if short if (rawData.at(i) <= 700) { div = 1; } else { div = 3; } // if long if (rawData.at(i+1) < 700) { divNext = 1; } else { divNext = 3; } // _ // | |___ = 0 -> in 4 delays => 1000 // _ // ___| | = 1 -> in 4 delays => 0001 if (div == 1 && divNext == 3) { binCode.append('0'); } else if(div == 3 && divNext == 1) { binCode.append('1'); } else { return; } } } else { return; } qCDebug(dcElro) << "Understands this protocol: " << binCode; if (binCode.left(20) == "00000100000000000001") { if (binCode.right(4) == "0100") { qCDebug(dcElro) << "Motion Detector OFF"; } else { qCDebug(dcElro) << "Motion Detector ON"; } } // get the channel of the remote signal (5 channels, true=1, false=0) QList group; for (int i = 1; i < 10; i+=2) { if (binCode.at(i-1) == '0' && binCode.at(i) == '1') { group << false; } else if(binCode.at(i-1) == '0' && binCode.at(i) == '0') { group << true; } else { return; } } // get the button letter QString button; QByteArray buttonCode = binCode.mid(10,10); if (buttonCode == "0001010101") { button = "A"; } else if (buttonCode == "0100010101") { button = "B"; } else if (buttonCode == "0101000101") { button = "C"; } else if(buttonCode == "0101010001") { button = "D"; } else if(buttonCode == "0101010100") { button = "E"; } else { return; } // get power status -> On = 0100, Off = 0001 bool power; if (binCode.right(4).toInt(0,2) == 1) { power = true; } else if(binCode.right(4).toInt(0,2) == 4) { power = false; } else { return; } qCDebug(dcElro) << group << buttonCode << power; }