/**************************************************************************** * * (c) 2009-2020 QGROUNDCONTROL PROJECT * * QGroundControl is licensed according to the terms in the file * COPYING.md in the root of the source code directory. * ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include "QGCApplication.h" #include "BluetoothLink.h" #include "QGC.h" #include "LinkManager.h" BluetoothLink::BluetoothLink(SharedLinkConfigurationPtr& config) : LinkInterface (config) { } BluetoothLink::~BluetoothLink() { disconnect(); #ifdef __ios__ if(_discoveryAgent) { _shutDown = true; _discoveryAgent->stop(); _discoveryAgent->deleteLater(); _discoveryAgent = nullptr; } #endif } void BluetoothLink::run() { } QString BluetoothLink::getName() const { return _config->name(); } void BluetoothLink::_writeBytes(const QByteArray bytes) { if (_targetSocket) { if(_targetSocket->write(bytes) > 0) { emit bytesSent(this, bytes); } else { qWarning() << "Bluetooth write error"; } } } void BluetoothLink::readBytes() { if (_targetSocket) { while (_targetSocket->bytesAvailable() > 0) { QByteArray datagram; datagram.resize(_targetSocket->bytesAvailable()); _targetSocket->read(datagram.data(), datagram.size()); emit bytesReceived(this, datagram); } } } void BluetoothLink::disconnect(void) { #ifdef __ios__ if(_discoveryAgent) { _shutDown = true; _discoveryAgent->stop(); _discoveryAgent->deleteLater(); _discoveryAgent = nullptr; } #endif if(_targetSocket) { // This prevents stale signals from calling the link after it has been deleted QObject::connect(_targetSocket, &QBluetoothSocket::readyRead, this, &BluetoothLink::readBytes); _targetSocket->deleteLater(); _targetSocket = nullptr; emit disconnected(); } _connectState = false; } bool BluetoothLink::_connect(void) { _hardwareConnect(); return true; } bool BluetoothLink::_hardwareConnect() { #ifdef __ios__ if(_discoveryAgent) { _shutDown = true; _discoveryAgent->stop(); _discoveryAgent->deleteLater(); _discoveryAgent = nullptr; } _discoveryAgent = new QBluetoothServiceDiscoveryAgent(this); QObject::connect(_discoveryAgent, &QBluetoothServiceDiscoveryAgent::serviceDiscovered, this, &BluetoothLink::serviceDiscovered); QObject::connect(_discoveryAgent, &QBluetoothServiceDiscoveryAgent::finished, this, &BluetoothLink::discoveryFinished); QObject::connect(_discoveryAgent, &QBluetoothServiceDiscoveryAgent::canceled, this, &BluetoothLink::discoveryFinished); _shutDown = false; _discoveryAgent->start(); #else _createSocket(); _targetSocket->connectToService(QBluetoothAddress(qobject_cast(_config.get())->device().address), QBluetoothUuid(QBluetoothUuid::SerialPort)); #endif return true; } void BluetoothLink::_createSocket() { if(_targetSocket) { delete _targetSocket; _targetSocket = nullptr; } _targetSocket = new QBluetoothSocket(QBluetoothServiceInfo::RfcommProtocol, this); QObject::connect(_targetSocket, &QBluetoothSocket::connected, this, &BluetoothLink::deviceConnected); QObject::connect(_targetSocket, &QBluetoothSocket::readyRead, this, &BluetoothLink::readBytes); QObject::connect(_targetSocket, &QBluetoothSocket::disconnected, this, &BluetoothLink::deviceDisconnected); QObject::connect(_targetSocket, static_cast(&QBluetoothSocket::error), this, &BluetoothLink::deviceError); } #ifdef __ios__ void BluetoothLink::serviceDiscovered(const QBluetoothServiceInfo& info) { if(!info.device().name().isEmpty() && !_targetSocket) { if(_config->device().uuid == info.device().deviceUuid() && _config->device().name == info.device().name()) { _createSocket(); _targetSocket->connectToService(info); } } } #endif #ifdef __ios__ void BluetoothLink::discoveryFinished() { if(_discoveryAgent && !_shutDown) { _shutDown = true; _discoveryAgent->deleteLater(); _discoveryAgent = nullptr; if(!_targetSocket) { _connectState = false; emit communicationError("Could not locate Bluetooth device:", _config->device().name); } } } #endif void BluetoothLink::deviceConnected() { _connectState = true; emit connected(); } void BluetoothLink::deviceDisconnected() { _connectState = false; qWarning() << "Bluetooth disconnected"; } void BluetoothLink::deviceError(QBluetoothSocket::SocketError error) { _connectState = false; qWarning() << "Bluetooth error" << error; emit communicationError(tr("Bluetooth Link Error"), _targetSocket->errorString()); } bool BluetoothLink::isConnected() const { return _connectState; } //-------------------------------------------------------------------------- //-- BluetoothConfiguration BluetoothConfiguration::BluetoothConfiguration(const QString& name) : LinkConfiguration(name) , _deviceDiscover(nullptr) { } BluetoothConfiguration::BluetoothConfiguration(BluetoothConfiguration* source) : LinkConfiguration(source) , _deviceDiscover(nullptr) , _device(source->device()) { } BluetoothConfiguration::~BluetoothConfiguration() { if(_deviceDiscover) { _deviceDiscover->stop(); delete _deviceDiscover; } } QString BluetoothConfiguration::settingsTitle() { if(qgcApp()->toolbox()->linkManager()->isBluetoothAvailable()) { return tr("Bluetooth Link Settings"); } else { return tr("Bluetooth Not Available"); } } void BluetoothConfiguration::copyFrom(LinkConfiguration *source) { LinkConfiguration::copyFrom(source); auto* usource = qobject_cast(source); Q_ASSERT(usource != nullptr); _device = usource->device(); } void BluetoothConfiguration::saveSettings(QSettings& settings, const QString& root) { settings.beginGroup(root); settings.setValue("deviceName", _device.name); #ifdef __ios__ settings.setValue("uuid", _device.uuid.toString()); #else settings.setValue("address",_device.address); #endif settings.endGroup(); } void BluetoothConfiguration::loadSettings(QSettings& settings, const QString& root) { settings.beginGroup(root); _device.name = settings.value("deviceName", _device.name).toString(); #ifdef __ios__ QString suuid = settings.value("uuid", _device.uuid.toString()).toString(); _device.uuid = QUuid(suuid); #else _device.address = settings.value("address", _device.address).toString(); #endif settings.endGroup(); } void BluetoothConfiguration::stopScan() { if(_deviceDiscover) { _deviceDiscover->stop(); _deviceDiscover->deleteLater(); _deviceDiscover = nullptr; emit scanningChanged(); } } void BluetoothConfiguration::startScan() { if(!_deviceDiscover) { _deviceDiscover = new QBluetoothDeviceDiscoveryAgent(this); connect(_deviceDiscover, &QBluetoothDeviceDiscoveryAgent::deviceDiscovered, this, &BluetoothConfiguration::deviceDiscovered); connect(_deviceDiscover, &QBluetoothDeviceDiscoveryAgent::finished, this, &BluetoothConfiguration::doneScanning); emit scanningChanged(); } else { _deviceDiscover->stop(); } _nameList.clear(); _deviceList.clear(); emit nameListChanged(); _deviceDiscover->setInquiryType(QBluetoothDeviceDiscoveryAgent::GeneralUnlimitedInquiry); _deviceDiscover->start(); } void BluetoothConfiguration::deviceDiscovered(QBluetoothDeviceInfo info) { if(!info.name().isEmpty() && info.isValid()) { #if 0 qDebug() << "Name: " << info.name(); qDebug() << "Address: " << info.address().toString(); qDebug() << "Service Classes:" << info.serviceClasses(); QList uuids = info.serviceUuids(); for (QBluetoothUuid uuid: uuids) { qDebug() << "Service UUID: " << uuid.toString(); } #endif BluetoothData data; data.name = info.name(); #ifdef __ios__ data.uuid = info.deviceUuid(); #else data.address = info.address().toString(); #endif if(!_deviceList.contains(data)) { _deviceList += data; _nameList += data.name; emit nameListChanged(); return; } } } void BluetoothConfiguration::doneScanning() { if(_deviceDiscover) { _deviceDiscover->deleteLater(); _deviceDiscover = nullptr; emit scanningChanged(); } } void BluetoothConfiguration::setDevName(const QString &name) { for(const BluetoothData& data: _deviceList) { if(data.name == name) { _device = data; emit devNameChanged(); #ifndef __ios__ emit addressChanged(); #endif return; } } } QString BluetoothConfiguration::address() { #ifdef __ios__ return {}; #else return _device.address; #endif }