remoteselector.cpp Example File

btfiletransfer/remoteselector.cpp
 /****************************************************************************
 **
 ** Copyright (C) 2017 The Qt Company Ltd.
 ** Contact: https://www.qt.io/licensing/
 **
 ** This file is part of the QtBluetooth module of the Qt Toolkit.
 **
 ** $QT_BEGIN_LICENSE:BSD$
 ** Commercial License Usage
 ** Licensees holding valid commercial Qt licenses may use this file in
 ** accordance with the commercial license agreement provided with the
 ** Software or, alternatively, in accordance with the terms contained in
 ** a written agreement between you and The Qt Company. For licensing terms
 ** and conditions see https://www.qt.io/terms-conditions. For further
 ** information use the contact form at https://www.qt.io/contact-us.
 **
 ** BSD License Usage
 ** Alternatively, you may use this file under the terms of the BSD license
 ** as follows:
 **
 ** "Redistribution and use in source and binary forms, with or without
 ** modification, are permitted provided that the following conditions are
 ** met:
 **   * Redistributions of source code must retain the above copyright
 **     notice, this list of conditions and the following disclaimer.
 **   * Redistributions in binary form must reproduce the above copyright
 **     notice, this list of conditions and the following disclaimer in
 **     the documentation and/or other materials provided with the
 **     distribution.
 **   * Neither the name of The Qt Company Ltd nor the names of its
 **     contributors may be used to endorse or promote products derived
 **     from this software without specific prior written permission.
 **
 **
 ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 ** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 ** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 ** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 ** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 ** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 ** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 ** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 ** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 ** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 ** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
 **
 ** $QT_END_LICENSE$
 **
 ****************************************************************************/

 #include "remoteselector.h"
 #include "ui_remoteselector.h"

 #include <qbluetoothdeviceinfo.h>
 #include <qbluetoothaddress.h>
 #include <qbluetoothtransferrequest.h>
 #include <qbluetoothtransferreply.h>
 #include <qbluetoothlocaldevice.h>

 #include <QMovie>
 #include <QMessageBox>
 #include <QFileDialog>
 #include <QCheckBox>

 #include "progress.h"
 #include "pindisplay.h"

 QT_USE_NAMESPACE

 RemoteSelector::RemoteSelector(QWidget *parent)
 :   QDialog(parent), ui(new Ui::RemoteSelector),
     m_localDevice(new QBluetoothLocalDevice), m_pindisplay(0),
     m_pairingError(false)
 {
     ui->setupUi(this);

     //Using default Bluetooth adapter
     QBluetoothAddress adapterAddress = m_localDevice->address();

     /*
      * In case of multiple Bluetooth adapters it is possible to
      * set which adapter will be used by providing MAC Address.
      * Example code:
      *
      * QBluetoothAddress adapterAddress("XX:XX:XX:XX:XX:XX");
      * m_discoveryAgent = new QBluetoothServiceDiscoveryAgent(adapterAddress);
      */

     m_discoveryAgent = new QBluetoothServiceDiscoveryAgent(adapterAddress);

     connect(m_discoveryAgent, SIGNAL(serviceDiscovered(QBluetoothServiceInfo)),
             this, SLOT(serviceDiscovered(QBluetoothServiceInfo)));
     connect(m_discoveryAgent, SIGNAL(finished()), this, SLOT(discoveryFinished()));
     connect(m_discoveryAgent, SIGNAL(canceled()), this, SLOT(discoveryFinished()));

     ui->remoteDevices->setColumnWidth(3, 75);
     ui->remoteDevices->setColumnWidth(4, 100);

     connect(m_localDevice, SIGNAL(pairingDisplayPinCode(QBluetoothAddress,QString)),
             this, SLOT(displayPin(QBluetoothAddress,QString)));
     connect(m_localDevice, SIGNAL(pairingDisplayConfirmation(QBluetoothAddress,QString)),
             this, SLOT(displayConfirmation(QBluetoothAddress,QString)));
     connect(m_localDevice, SIGNAL(pairingFinished(QBluetoothAddress,QBluetoothLocalDevice::Pairing)),
             this, SLOT(pairingFinished(QBluetoothAddress,QBluetoothLocalDevice::Pairing)));
     connect(m_localDevice, SIGNAL(error(QBluetoothLocalDevice::Error)),
             this, SLOT(pairingError(QBluetoothLocalDevice::Error)));

     ui->busyWidget->setMovie(new QMovie(":/icons/busy.gif"));
     ui->busyWidget->movie()->start();

     ui->pairingBusy->setMovie(new QMovie(":/icons/pairing.gif"));
     ui->pairingBusy->hide();

     ui->remoteDevices->clearContents();
     ui->remoteDevices->setRowCount(0);
 }

 RemoteSelector::~RemoteSelector()
 {
     delete ui;
     delete m_discoveryAgent;
     delete m_localDevice;
 }

 void RemoteSelector::startDiscovery(const QBluetoothUuid &uuid)
 {
     ui->stopButton->setDisabled(false);
     if (m_discoveryAgent->isActive())
         m_discoveryAgent->stop();

     m_discoveryAgent->setUuidFilter(uuid);
     m_discoveryAgent->start();

     if (!m_discoveryAgent->isActive() ||
             m_discoveryAgent->error() != QBluetoothServiceDiscoveryAgent::NoError) {
         ui->status->setText(tr("Cannot find remote services."));
     } else {
         ui->status->setText(tr("Scanning..."));
         ui->busyWidget->show();
         ui->busyWidget->movie()->start();
     }
 }

 QBluetoothServiceInfo RemoteSelector::service() const
 {
     return m_service;
 }

 void RemoteSelector::serviceDiscovered(const QBluetoothServiceInfo &serviceInfo)
 {
 #if 0
     qDebug() << "Discovered service on"
              << serviceInfo.device().name() << serviceInfo.device().address().toString();
     qDebug() << "\tService name:" << serviceInfo.serviceName();
     qDebug() << "\tDescription:"
              << serviceInfo.attribute(QBluetoothServiceInfo::ServiceDescription).toString();
     qDebug() << "\tProvider:"
              << serviceInfo.attribute(QBluetoothServiceInfo::ServiceProvider).toString();
     qDebug() << "\tL2CAP protocol service multiplexer:"
              << serviceInfo.protocolServiceMultiplexer();
     qDebug() << "\tRFCOMM server channel:" << serviceInfo.serverChannel();
 #endif

     QString remoteName;
     if (serviceInfo.device().name().isEmpty())
         remoteName = serviceInfo.device().address().toString();
     else
         remoteName = serviceInfo.device().name();

 //    QListWidgetItem *item =
 //        new QListWidgetItem(QString::fromLatin1("%1\t%2\t%3").arg(serviceInfo.device().address().toString(),
 //                                                             serviceInfo.device().name(), serviceInfo.serviceName()));

     QMutableMapIterator<int, QBluetoothServiceInfo> i(m_discoveredServices);
     while (i.hasNext()){
         i.next();
         if (serviceInfo.device().address() == i.value().device().address()){
             i.setValue(serviceInfo);
             return;
         }
     }

     int row = ui->remoteDevices->rowCount();
     ui->remoteDevices->insertRow(row);
     QTableWidgetItem *item = new QTableWidgetItem(serviceInfo.device().address().toString());
     ui->remoteDevices->setItem(row, 0, item);
     item = new QTableWidgetItem(serviceInfo.device().name());
     ui->remoteDevices->setItem(row, 1, item);
     item = new QTableWidgetItem(serviceInfo.serviceName());

     ui->remoteDevices->setItem(row, 2, item);

     QBluetoothLocalDevice::Pairing p;

     p = m_localDevice->pairingStatus(serviceInfo.device().address());

     ui->remoteDevices->blockSignals(true);

     item = new QTableWidgetItem();
     if ((p&QBluetoothLocalDevice::Paired) || (p&QBluetoothLocalDevice::AuthorizedPaired))
         item->setCheckState(Qt::Checked);
     else
         item->setCheckState(Qt::Unchecked);
     ui->remoteDevices->setItem(row, 3, item);

     item = new QTableWidgetItem();
     if (p&QBluetoothLocalDevice::AuthorizedPaired)
         item->setCheckState(Qt::Checked);
     else
         item->setCheckState(Qt::Unchecked);

     ui->remoteDevices->setItem(row, 4, item);

     ui->remoteDevices->blockSignals(false);

     m_discoveredServices.insert(row, serviceInfo);
 }

 void RemoteSelector::discoveryFinished()
 {
     ui->status->setText(tr("Select the device to send to."));
     ui->stopButton->setDisabled(true);
     ui->busyWidget->movie()->stop();
     ui->busyWidget->hide();
 }

 void RemoteSelector::startDiscovery()
 {
     startDiscovery(QBluetoothUuid(QBluetoothUuid::ObexObjectPush));
 }

 void RemoteSelector::on_refreshPB_clicked()
 {
     startDiscovery();
     ui->stopButton->setDisabled(false);
 }

 void RemoteSelector::on_fileSelectPB_clicked()
 {
     ui->fileName->setText(QFileDialog::getOpenFileName());
     if (m_service.isValid())
         ui->sendButton->setDisabled(false);
 }

 void RemoteSelector::on_sendButton_clicked()
 {
     QBluetoothTransferManager mgr;
     QBluetoothTransferRequest req(m_service.device().address());

     m_file = new QFile(ui->fileName->text());

     Progress *p = new Progress;
     p->setStatus("Sending to: " + m_service.device().name(), "Waiting for start");
     p->show();

     QBluetoothTransferReply *reply = mgr.put(req, m_file);
     //mgr is default parent
     //ensure that mgr doesn't take reply down when leaving scope
     reply->setParent(this);
     if (reply->error()){
         qDebug() << "Failed to send file";
         p->finished(reply);
         reply->deleteLater();
         return;
     }

     connect(reply, SIGNAL(transferProgress(qint64,qint64)), p, SLOT(uploadProgress(qint64,qint64)));
     connect(reply, SIGNAL(finished(QBluetoothTransferReply*)), p, SLOT(finished(QBluetoothTransferReply*)));
     connect(p, SIGNAL(rejected()), reply, SLOT(abort()));
 }

 void RemoteSelector::on_stopButton_clicked()
 {
     m_discoveryAgent->stop();
 }

 QString RemoteSelector::addressToName(const QBluetoothAddress &address)
 {
     QMapIterator<int, QBluetoothServiceInfo> i(m_discoveredServices);
     while (i.hasNext()){
         i.next();
         if (i.value().device().address() == address)
             return i.value().device().name();
     }
     return address.toString();
 }

 void RemoteSelector::displayPin(const QBluetoothAddress &address, QString pin)
 {
     if (m_pindisplay)
         m_pindisplay->deleteLater();
     m_pindisplay = new pinDisplay(QString("Enter pairing pin on: %1").arg(addressToName(address)), pin, this);
     m_pindisplay->show();
 }

 void RemoteSelector::displayConfirmation(const QBluetoothAddress &address, QString pin)
 {
     Q_UNUSED(address);

     if (m_pindisplay)
         m_pindisplay->deleteLater();
     m_pindisplay = new pinDisplay(QString("Confirm this pin is the same"), pin, this);
     connect(m_pindisplay, SIGNAL(accepted()), this, SLOT(displayConfAccepted()));
     connect(m_pindisplay, SIGNAL(rejected()), this, SLOT(displayConfReject()));
     m_pindisplay->setOkCancel();
     m_pindisplay->show();
 }

 void RemoteSelector::displayConfAccepted()
 {
     m_localDevice->pairingConfirmation(true);
 }
 void RemoteSelector::displayConfReject()
 {
     m_localDevice->pairingConfirmation(false);
 }

 void RemoteSelector::pairingFinished(const QBluetoothAddress &address, QBluetoothLocalDevice::Pairing status)
 {
     QBluetoothServiceInfo service;
     int row = 0;

     ui->pairingBusy->hide();
     ui->pairingBusy->movie()->stop();

     ui->remoteDevices->blockSignals(true);

     for (int i = 0; i < m_discoveredServices.count(); i++){
         if (m_discoveredServices.value(i).device().address() == address){
             service = m_discoveredServices.value(i);
             row = i;
             break;
         }
     }

     if (m_pindisplay)
         delete m_pindisplay;

     QMessageBox msgBox;
     if (m_pairingError) {
         msgBox.setText("Pairing failed with " + address.toString());
     } else if (status == QBluetoothLocalDevice::Paired
                || status == QBluetoothLocalDevice::AuthorizedPaired) {
         msgBox.setText("Paired successfully with " + address.toString());
     } else {
         msgBox.setText("Pairing released with " + address.toString());
     }

     if (service.isValid()){
         if (status == QBluetoothLocalDevice::AuthorizedPaired){
             ui->remoteDevices->item(row, 3)->setCheckState(Qt::Checked);
             ui->remoteDevices->item(row, 4)->setCheckState(Qt::Checked);
         }
         else if (status == QBluetoothLocalDevice::Paired){
             ui->remoteDevices->item(row, 3)->setCheckState(Qt::Checked);
             ui->remoteDevices->item(row, 4)->setCheckState(Qt::Unchecked);
         }
         else {
             ui->remoteDevices->item(row, 3)->setCheckState(Qt::Unchecked);
             ui->remoteDevices->item(row, 4)->setCheckState(Qt::Unchecked);
         }
     }

     m_pairingError = false;
     msgBox.exec();

     ui->remoteDevices->blockSignals(false);
 }

 void RemoteSelector::pairingError(QBluetoothLocalDevice::Error error)
 {
     if (error != QBluetoothLocalDevice::PairingError)
         return;

     m_pairingError = true;
     pairingFinished(m_service.device().address(), QBluetoothLocalDevice::Unpaired);
 }

 void RemoteSelector::on_remoteDevices_cellClicked(int row, int column)
 {
     Q_UNUSED(column);

     m_service = m_discoveredServices.value(row);
     if (!ui->fileName->text().isEmpty()) {
         ui->sendButton->setDisabled(false);
     }
 }

 void RemoteSelector::on_remoteDevices_itemChanged(QTableWidgetItem* item)
 {
     int row = item->row();
     int column = item->column();
     m_service = m_discoveredServices.value(row);

     if (column < 3)
         return;

     if (item->checkState() == Qt::Unchecked && column == 3){
         m_localDevice->requestPairing(m_service.device().address(), QBluetoothLocalDevice::Unpaired);
         return; // don't continue and start movie
     }
     else if ((item->checkState() == Qt::Checked && column == 3) ||
             (item->checkState() == Qt::Unchecked && column == 4)){
         m_localDevice->requestPairing(m_service.device().address(), QBluetoothLocalDevice::Paired);
         ui->remoteDevices->blockSignals(true);
         ui->remoteDevices->item(row, column)->setCheckState(Qt::PartiallyChecked);
         ui->remoteDevices->blockSignals(false);
     }
     else if (item->checkState() == Qt::Checked && column == 4){
         m_localDevice->requestPairing(m_service.device().address(), QBluetoothLocalDevice::AuthorizedPaired);
         ui->remoteDevices->blockSignals(true);
         ui->remoteDevices->item(row, column)->setCheckState(Qt::PartiallyChecked);
         ui->remoteDevices->blockSignals(false);
     }
     ui->pairingBusy->show();
     ui->pairingBusy->movie()->start();
 }