348 lines
11 KiB
C++
348 lines
11 KiB
C++
/**
|
|
* SPDX-FileCopyrightText: 2013 Albert Vaca <albertvaka@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
|
*/
|
|
|
|
#include "kcm.h"
|
|
|
|
#include <KAboutData>
|
|
#include <KColorSchemeManager>
|
|
#include <KConfigGroup>
|
|
#include <KLocalizedString>
|
|
#include <KPluginFactory>
|
|
#include <KPluginMetaData>
|
|
#include <kcmutils_version.h>
|
|
|
|
#include <QQmlContext>
|
|
#include <QQuickItem>
|
|
#include <QQuickStyle>
|
|
|
|
#include "dbushelpers.h"
|
|
#include "dbusinterfaces.h"
|
|
#include "devicesmodel.h"
|
|
#include "devicessortproxymodel.h"
|
|
#include "kdeconnect-version.h"
|
|
|
|
K_PLUGIN_CLASS_WITH_JSON(KdeConnectKcm, "kcm_kdeconnect.json")
|
|
|
|
class QQuickWidgetPaleteChangeWatcher : public QObject
|
|
{
|
|
using QObject::QObject;
|
|
|
|
bool eventFilter(QObject *watched, QEvent *event) override
|
|
{
|
|
if (event->type() == QEvent::PaletteChange || event->type() == QEvent::ApplicationPaletteChange) {
|
|
// We know that watched is a QQuickWidget
|
|
QQuickWidget *w = static_cast<QQuickWidget *>(watched);
|
|
w->setClearColor(w->palette().color(QPalette::Window));
|
|
}
|
|
return QObject::eventFilter(watched, event);
|
|
}
|
|
};
|
|
|
|
KdeConnectKcm::KdeConnectKcm(QObject *parent, const KPluginMetaData &md, const QVariantList &args)
|
|
: KCModule(parent, md)
|
|
, daemon(new DaemonDbusInterface(this))
|
|
, devicesModel(new DevicesModel(this))
|
|
, currentDevice(nullptr)
|
|
{
|
|
#ifdef Q_OS_WIN
|
|
KColorSchemeManager::instance();
|
|
QApplication::setStyle(QStringLiteral("breeze"));
|
|
#endif
|
|
|
|
// Default to org.kde.desktop style unless the user forces another style
|
|
if (qEnvironmentVariableIsEmpty("QT_QUICK_CONTROLS_STYLE")) {
|
|
QQuickStyle::setStyle(QStringLiteral("org.kde.desktop"));
|
|
}
|
|
|
|
kcmUi.setupUi(widget());
|
|
|
|
sortProxyModel = new DevicesSortProxyModel(devicesModel);
|
|
|
|
kcmUi.list_quick_widget->setMinimumWidth(250);
|
|
kcmUi.list_quick_widget->rootContext()->setContextObject(new KLocalizedContext(kcmUi.list_quick_widget));
|
|
kcmUi.list_quick_widget->setClearColor(kcmUi.list_quick_widget->palette().color(QPalette::Window));
|
|
kcmUi.list_quick_widget->setSource(QUrl(QStringLiteral("qrc:/kdeconnectkcm/list.qml")));
|
|
kcmUi.list_quick_widget->rootObject()->setProperty("model", QVariant::fromValue(sortProxyModel));
|
|
connect(kcmUi.list_quick_widget->rootObject(), SIGNAL(clicked(QString)), this, SLOT(deviceSelected(QString)));
|
|
|
|
kcmUi.list_quick_widget->installEventFilter(new QQuickWidgetPaleteChangeWatcher(kcmUi.list_quick_widget));
|
|
|
|
kcmUi.deviceInfo->setVisible(false);
|
|
kcmUi.progressBar->setVisible(false);
|
|
kcmUi.messages->setVisible(false);
|
|
|
|
// Workaround: If we set this directly (or if we set it in the .ui file), the layout breaks
|
|
kcmUi.noDeviceLinks->setWordWrap(false);
|
|
QTimer::singleShot(0, this, [this] {
|
|
kcmUi.noDeviceLinks->setWordWrap(true);
|
|
});
|
|
|
|
setWhenAvailable(
|
|
daemon->announcedName(),
|
|
[this](bool error, const QString &announcedName) {
|
|
kcmUi.renameShow_button->setEnabled(!error);
|
|
if (error) {
|
|
kcmUi.rename_label->setText(i18n("Error: KDE Connect is not running"));
|
|
} else {
|
|
kcmUi.rename_label->setText(announcedName);
|
|
kcmUi.rename_edit->setText(announcedName);
|
|
}
|
|
},
|
|
this);
|
|
|
|
kcmUi.bluetoothEnabled_checkbox->setChecked(false);
|
|
|
|
setWhenAvailable(
|
|
daemon->linkProviders(),
|
|
[this](bool error, const QStringList linkProviders) {
|
|
for (int i = 0; i < linkProviders.size(); ++i) {
|
|
QString linkProvider = QString(linkProviders.at(i).constData());
|
|
qDebug() << "link provider: " << linkProvider;
|
|
|
|
if (linkProvider == QStringLiteral("bluetooth")) {
|
|
kcmUi.bluetoothEnabled_checkbox->setChecked(true);
|
|
}
|
|
}
|
|
},
|
|
this);
|
|
|
|
connect(daemon, &DaemonDbusInterface::announcedNameChanged, kcmUi.rename_edit, &QLineEdit::setText);
|
|
connect(daemon, &DaemonDbusInterface::announcedNameChanged, kcmUi.rename_label, &QLabel::setText);
|
|
setRenameMode(false);
|
|
|
|
setButtons(KCModule::Help | KCModule::NoAdditionalButton);
|
|
|
|
connect(kcmUi.accept_button, &QAbstractButton::clicked, this, &KdeConnectKcm::acceptPairing);
|
|
connect(kcmUi.reject_button, &QAbstractButton::clicked, this, &KdeConnectKcm::cancelPairing);
|
|
connect(kcmUi.cancel_button, &QAbstractButton::clicked, this, &KdeConnectKcm::cancelPairing);
|
|
connect(kcmUi.pair_button, &QAbstractButton::clicked, this, &KdeConnectKcm::requestPairing);
|
|
connect(kcmUi.unpair_button, &QAbstractButton::clicked, this, &KdeConnectKcm::unpair);
|
|
connect(kcmUi.ping_button, &QAbstractButton::clicked, this, &KdeConnectKcm::sendPing);
|
|
connect(kcmUi.refresh_button, &QAbstractButton::clicked, this, &KdeConnectKcm::refresh);
|
|
connect(kcmUi.rename_edit, &QLineEdit::returnPressed, this, &KdeConnectKcm::renameDone);
|
|
connect(kcmUi.renameDone_button, &QAbstractButton::clicked, this, &KdeConnectKcm::renameDone);
|
|
connect(kcmUi.renameShow_button, &QAbstractButton::clicked, this, &KdeConnectKcm::renameShow);
|
|
connect(kcmUi.pluginSelector, &KPluginWidget::changed, this, &KdeConnectKcm::pluginsConfigChanged);
|
|
|
|
if (!args.isEmpty() && !args.first().isNull() && args.first().canConvert<QString>()) {
|
|
const QString input = args.first().toString();
|
|
const auto colonIdx = input.indexOf(QLatin1Char(':'));
|
|
const QString deviceId = input.left(colonIdx);
|
|
const QString pluginCM = colonIdx < 0 ? QString() : input.mid(colonIdx + 1);
|
|
|
|
connect(devicesModel, &DevicesModel::rowsInserted, this, [this, deviceId, pluginCM]() {
|
|
kcmUi.list_quick_widget->rootObject()->setProperty("currentDeviceId", deviceId);
|
|
deviceSelected(deviceId);
|
|
if (!pluginCM.isEmpty()) {
|
|
kcmUi.pluginSelector->showConfiguration(pluginCM);
|
|
}
|
|
disconnect(devicesModel, &DevicesModel::rowsInserted, this, nullptr);
|
|
});
|
|
}
|
|
}
|
|
|
|
void KdeConnectKcm::renameShow()
|
|
{
|
|
setRenameMode(true);
|
|
}
|
|
|
|
void KdeConnectKcm::renameDone()
|
|
{
|
|
QString newName = kcmUi.rename_edit->text();
|
|
if (newName.isEmpty()) {
|
|
// Rollback changes
|
|
kcmUi.rename_edit->setText(kcmUi.rename_label->text());
|
|
} else {
|
|
kcmUi.rename_label->setText(newName);
|
|
daemon->setAnnouncedName(newName);
|
|
}
|
|
setRenameMode(false);
|
|
}
|
|
|
|
void KdeConnectKcm::setRenameMode(bool b)
|
|
{
|
|
kcmUi.renameDone_button->setVisible(b);
|
|
kcmUi.rename_edit->setVisible(b);
|
|
kcmUi.renameShow_button->setVisible(!b);
|
|
kcmUi.rename_label->setVisible(!b);
|
|
}
|
|
|
|
KdeConnectKcm::~KdeConnectKcm()
|
|
{
|
|
}
|
|
|
|
void KdeConnectKcm::refresh()
|
|
{
|
|
daemon->forceOnNetworkChange();
|
|
}
|
|
|
|
void KdeConnectKcm::deviceSelected(const QString &deviceId)
|
|
{
|
|
if (currentDevice) {
|
|
disconnect(currentDevice, nullptr, this, nullptr);
|
|
}
|
|
|
|
currentDevice = devicesModel->getDevice(devicesModel->rowForDevice(deviceId));
|
|
if (!currentDevice) {
|
|
currentDevice = nullptr;
|
|
kcmUi.deviceInfo->setVisible(false);
|
|
return;
|
|
}
|
|
|
|
kcmUi.noDevicePlaceholder->setVisible(false);
|
|
bool valid = (currentDevice != nullptr && currentDevice->isValid());
|
|
kcmUi.deviceInfo->setVisible(valid);
|
|
if (!valid) {
|
|
return;
|
|
}
|
|
|
|
kcmUi.messages->setVisible(false);
|
|
resetDeviceView();
|
|
|
|
connect(currentDevice, &DeviceDbusInterface::pluginsChanged, this, &KdeConnectKcm::resetCurrentDevice);
|
|
connect(currentDevice, &DeviceDbusInterface::pairingFailed, this, &KdeConnectKcm::pairingFailed);
|
|
connect(currentDevice, &DeviceDbusInterface::pairStateChanged, this, &KdeConnectKcm::setCurrentDevicePairState);
|
|
}
|
|
|
|
void KdeConnectKcm::resetCurrentDevice()
|
|
{
|
|
const QStringList supportedPluginNames = currentDevice->supportedPlugins();
|
|
|
|
if (m_oldSupportedPluginNames != supportedPluginNames) {
|
|
resetDeviceView();
|
|
}
|
|
}
|
|
|
|
void KdeConnectKcm::resetDeviceView()
|
|
{
|
|
kcmUi.verificationKey->setText(i18n("Key: %1", currentDevice->verificationKey()));
|
|
|
|
kcmUi.name_label->setText(currentDevice->name());
|
|
setWhenAvailable(
|
|
currentDevice->pairStateAsInt(),
|
|
[this](bool error, int pairStateAsInt) {
|
|
if (!error) {
|
|
setCurrentDevicePairState(pairStateAsInt);
|
|
}
|
|
},
|
|
this);
|
|
|
|
const QVector<KPluginMetaData> pluginInfo = KPluginMetaData::findPlugins(QStringLiteral("kdeconnect"));
|
|
QVector<KPluginMetaData> availablePluginInfo;
|
|
|
|
m_oldSupportedPluginNames = currentDevice->supportedPlugins();
|
|
for (auto it = pluginInfo.cbegin(), itEnd = pluginInfo.cend(); it != itEnd; ++it) {
|
|
if (m_oldSupportedPluginNames.contains(it->pluginId())) {
|
|
availablePluginInfo.append(*it);
|
|
}
|
|
}
|
|
|
|
KSharedConfigPtr deviceConfig = KSharedConfig::openConfig(currentDevice->pluginsConfigFile());
|
|
kcmUi.pluginSelector->clear();
|
|
kcmUi.pluginSelector->setConfigurationArguments({currentDevice->id()});
|
|
kcmUi.pluginSelector->addPlugins(availablePluginInfo, i18n("Available plugins"));
|
|
kcmUi.pluginSelector->setConfig(deviceConfig->group(QStringLiteral("Plugins")));
|
|
}
|
|
|
|
void KdeConnectKcm::requestPairing()
|
|
{
|
|
if (!currentDevice) {
|
|
return;
|
|
}
|
|
|
|
kcmUi.messages->hide();
|
|
|
|
currentDevice->requestPairing();
|
|
}
|
|
|
|
void KdeConnectKcm::unpair()
|
|
{
|
|
if (!currentDevice) {
|
|
return;
|
|
}
|
|
|
|
currentDevice->unpair();
|
|
}
|
|
|
|
void KdeConnectKcm::acceptPairing()
|
|
{
|
|
if (!currentDevice) {
|
|
return;
|
|
}
|
|
|
|
currentDevice->acceptPairing();
|
|
}
|
|
|
|
void KdeConnectKcm::cancelPairing()
|
|
{
|
|
if (!currentDevice) {
|
|
return;
|
|
}
|
|
|
|
currentDevice->cancelPairing();
|
|
}
|
|
|
|
void KdeConnectKcm::pairingFailed(const QString &error)
|
|
{
|
|
if (sender() != currentDevice)
|
|
return;
|
|
|
|
kcmUi.messages->setText(i18n("Error trying to pair: %1", error));
|
|
kcmUi.messages->animatedShow();
|
|
}
|
|
|
|
void KdeConnectKcm::setCurrentDevicePairState(int pairStateAsInt)
|
|
{
|
|
PairState state = (PairState)pairStateAsInt; // Hack because qdbus doesn't like enums
|
|
kcmUi.accept_button->setVisible(state == PairState::RequestedByPeer);
|
|
kcmUi.reject_button->setVisible(state == PairState::RequestedByPeer);
|
|
kcmUi.cancel_button->setVisible(state == PairState::Requested);
|
|
kcmUi.pair_button->setVisible(state == PairState::NotPaired);
|
|
kcmUi.unpair_button->setVisible(state == PairState::Paired);
|
|
kcmUi.progressBar->setVisible(state == PairState::Requested);
|
|
kcmUi.ping_button->setVisible(state == PairState::Paired);
|
|
switch (state) {
|
|
case PairState::Paired:
|
|
kcmUi.status_label->setText(i18n("(paired)"));
|
|
break;
|
|
case PairState::NotPaired:
|
|
kcmUi.status_label->setText(i18n("(not paired)"));
|
|
break;
|
|
case PairState::RequestedByPeer:
|
|
kcmUi.status_label->setText(i18n("(incoming pair request)"));
|
|
break;
|
|
case PairState::Requested:
|
|
kcmUi.status_label->setText(i18n("(pairing requested)"));
|
|
break;
|
|
}
|
|
}
|
|
|
|
void KdeConnectKcm::pluginsConfigChanged(bool changed)
|
|
{
|
|
if (!changed)
|
|
return;
|
|
|
|
if (!currentDevice)
|
|
return;
|
|
|
|
kcmUi.pluginSelector->save();
|
|
currentDevice->reloadPlugins();
|
|
}
|
|
|
|
void KdeConnectKcm::save()
|
|
{
|
|
KCModule::save();
|
|
}
|
|
|
|
void KdeConnectKcm::sendPing()
|
|
{
|
|
if (!currentDevice)
|
|
return;
|
|
currentDevice->pluginCall(QStringLiteral("ping"), QStringLiteral("sendPing"));
|
|
}
|
|
|
|
#include "kcm.moc"
|
|
#include "moc_kcm.cpp"
|