mirror of
https://github.com/Qortal/Brooklyn.git
synced 2025-02-23 15:45:53 +00:00
273 lines
9.0 KiB
C++
273 lines
9.0 KiB
C++
|
/*
|
||
|
SPDX-FileCopyrightText: 2007 Menard Alexis <darktears31@gmail.com>
|
||
|
|
||
|
SPDX-License-Identifier: LGPL-2.0-or-later
|
||
|
*/
|
||
|
|
||
|
#include "hotplugengine.h"
|
||
|
#include "hotplugservice.h"
|
||
|
|
||
|
#include <QDir>
|
||
|
#include <QDirIterator>
|
||
|
#include <QStandardPaths>
|
||
|
#include <QTimer>
|
||
|
|
||
|
#include <KConfigGroup>
|
||
|
#include <KDesktopFile>
|
||
|
#include <KDirWatch>
|
||
|
#include <Plasma/DataContainer>
|
||
|
#include <QDebug>
|
||
|
#include <kdesktopfileactions.h>
|
||
|
|
||
|
// solid specific includes
|
||
|
#include <Solid/Device>
|
||
|
#include <Solid/DeviceInterface>
|
||
|
#include <Solid/DeviceNotifier>
|
||
|
#include <Solid/OpticalDisc>
|
||
|
#include <Solid/StorageDrive>
|
||
|
#include <Solid/StorageVolume>
|
||
|
|
||
|
//#define HOTPLUGENGINE_TIMING
|
||
|
|
||
|
HotplugEngine::HotplugEngine(QObject *parent, const QVariantList &args)
|
||
|
: Plasma::DataEngine(parent, args)
|
||
|
, m_dirWatch(new KDirWatch(this))
|
||
|
{
|
||
|
const QStringList folders =
|
||
|
QStandardPaths::locateAll(QStandardPaths::GenericDataLocation, QStringLiteral("solid/actions"), QStandardPaths::LocateDirectory);
|
||
|
|
||
|
for (const QString &folder : folders) {
|
||
|
m_dirWatch->addDir(folder, KDirWatch::WatchFiles);
|
||
|
}
|
||
|
connect(m_dirWatch, &KDirWatch::created, this, &HotplugEngine::updatePredicates);
|
||
|
connect(m_dirWatch, &KDirWatch::deleted, this, &HotplugEngine::updatePredicates);
|
||
|
connect(m_dirWatch, &KDirWatch::dirty, this, &HotplugEngine::updatePredicates);
|
||
|
init();
|
||
|
}
|
||
|
|
||
|
HotplugEngine::~HotplugEngine()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void HotplugEngine::init()
|
||
|
{
|
||
|
findPredicates();
|
||
|
|
||
|
Solid::Predicate p(Solid::DeviceInterface::StorageAccess);
|
||
|
p |= Solid::Predicate(Solid::DeviceInterface::StorageDrive);
|
||
|
p |= Solid::Predicate(Solid::DeviceInterface::StorageVolume);
|
||
|
p |= Solid::Predicate(Solid::DeviceInterface::OpticalDrive);
|
||
|
p |= Solid::Predicate(Solid::DeviceInterface::OpticalDisc);
|
||
|
p |= Solid::Predicate(Solid::DeviceInterface::PortableMediaPlayer);
|
||
|
p |= Solid::Predicate(Solid::DeviceInterface::Camera);
|
||
|
const QList<Solid::Device> devices = Solid::Device::listFromQuery(p);
|
||
|
for (const Solid::Device &dev : devices) {
|
||
|
m_startList.insert(dev.udi(), dev);
|
||
|
}
|
||
|
|
||
|
connect(Solid::DeviceNotifier::instance(), &Solid::DeviceNotifier::deviceAdded, this, &HotplugEngine::onDeviceAdded);
|
||
|
connect(Solid::DeviceNotifier::instance(), &Solid::DeviceNotifier::deviceRemoved, this, &HotplugEngine::onDeviceRemoved);
|
||
|
|
||
|
m_encryptedPredicate = Solid::Predicate(QStringLiteral("StorageVolume"), QStringLiteral("usage"), "Encrypted");
|
||
|
|
||
|
processNextStartupDevice();
|
||
|
}
|
||
|
|
||
|
Plasma::Service *HotplugEngine::serviceForSource(const QString &source)
|
||
|
{
|
||
|
return new HotplugService(this, source);
|
||
|
}
|
||
|
|
||
|
void HotplugEngine::processNextStartupDevice()
|
||
|
{
|
||
|
if (!m_startList.isEmpty()) {
|
||
|
QHash<QString, Solid::Device>::iterator it = m_startList.begin();
|
||
|
// Solid::Device dev = const_cast<Solid::Device &>(m_startList.takeFirst());
|
||
|
handleDeviceAdded(it.value(), false);
|
||
|
m_startList.erase(it);
|
||
|
}
|
||
|
|
||
|
if (m_startList.isEmpty()) {
|
||
|
m_predicates.clear();
|
||
|
} else {
|
||
|
QTimer::singleShot(0, this, &HotplugEngine::processNextStartupDevice);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void HotplugEngine::findPredicates()
|
||
|
{
|
||
|
m_predicates.clear();
|
||
|
QStringList files;
|
||
|
const QStringList dirs = QStandardPaths::locateAll(QStandardPaths::GenericDataLocation, QStringLiteral("solid/actions"), QStandardPaths::LocateDirectory);
|
||
|
for (const QString &dir : dirs) {
|
||
|
QDirIterator it(dir, QStringList() << QStringLiteral("*.desktop"));
|
||
|
while (it.hasNext()) {
|
||
|
files.prepend(it.next());
|
||
|
}
|
||
|
}
|
||
|
// qDebug() << files;
|
||
|
for (const QString &path : qAsConst(files)) {
|
||
|
KDesktopFile cfg(path);
|
||
|
const QString string_predicate = cfg.desktopGroup().readEntry("X-KDE-Solid-Predicate");
|
||
|
// qDebug() << path << string_predicate;
|
||
|
m_predicates.insert(QUrl(path).fileName(), Solid::Predicate::fromString(string_predicate));
|
||
|
}
|
||
|
|
||
|
if (m_predicates.isEmpty()) {
|
||
|
m_predicates.insert(QString(), Solid::Predicate::fromString(QString()));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void HotplugEngine::updatePredicates(const QString &path)
|
||
|
{
|
||
|
Q_UNUSED(path)
|
||
|
|
||
|
findPredicates();
|
||
|
|
||
|
QHashIterator<QString, Solid::Device> it(m_devices);
|
||
|
while (it.hasNext()) {
|
||
|
it.next();
|
||
|
Solid::Device device(it.value());
|
||
|
QString udi(it.key());
|
||
|
|
||
|
const QStringList predicates = predicatesForDevice(device);
|
||
|
if (!predicates.isEmpty()) {
|
||
|
if (sources().contains(udi)) {
|
||
|
Plasma::DataEngine::Data data;
|
||
|
data.insert(QStringLiteral("predicateFiles"), predicates);
|
||
|
data.insert(QStringLiteral("actions"), actionsForPredicates(predicates));
|
||
|
setData(udi, data);
|
||
|
} else {
|
||
|
handleDeviceAdded(device, false);
|
||
|
}
|
||
|
} else if (!m_encryptedPredicate.matches(device) && sources().contains(udi)) {
|
||
|
removeSource(udi);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
QStringList HotplugEngine::predicatesForDevice(Solid::Device &device) const
|
||
|
{
|
||
|
QStringList interestingDesktopFiles;
|
||
|
// search in all desktop configuration file if the device inserted is a correct device
|
||
|
QHashIterator<QString, Solid::Predicate> it(m_predicates);
|
||
|
// qDebug() << "=================" << udi;
|
||
|
while (it.hasNext()) {
|
||
|
it.next();
|
||
|
if (it.value().matches(device)) {
|
||
|
// qDebug() << " hit" << it.key();
|
||
|
interestingDesktopFiles << it.key();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return interestingDesktopFiles;
|
||
|
}
|
||
|
|
||
|
QVariantList HotplugEngine::actionsForPredicates(const QStringList &predicates) const
|
||
|
{
|
||
|
QVariantList actions;
|
||
|
actions.reserve(predicates.count());
|
||
|
|
||
|
for (const QString &desktop : predicates) {
|
||
|
const QString actionUrl = QStandardPaths::locate(QStandardPaths::GenericDataLocation, "solid/actions/" + desktop);
|
||
|
QList<KServiceAction> services = KDesktopFileActions::userDefinedServices(KService(actionUrl), true);
|
||
|
if (!services.isEmpty()) {
|
||
|
Plasma::DataEngine::Data action;
|
||
|
action.insert(QStringLiteral("predicate"), desktop);
|
||
|
action.insert(QStringLiteral("text"), services[0].text());
|
||
|
action.insert(QStringLiteral("icon"), services[0].icon());
|
||
|
actions << action;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return actions;
|
||
|
}
|
||
|
|
||
|
void HotplugEngine::onDeviceAdded(const QString &udi)
|
||
|
{
|
||
|
Solid::Device device(udi);
|
||
|
handleDeviceAdded(device);
|
||
|
}
|
||
|
|
||
|
void HotplugEngine::handleDeviceAdded(Solid::Device &device, bool added)
|
||
|
{
|
||
|
// qDebug() << "adding" << device.udi();
|
||
|
#ifdef HOTPLUGENGINE_TIMING
|
||
|
QTime t;
|
||
|
t.start();
|
||
|
#endif
|
||
|
// Skip things we know we don't care about
|
||
|
if (device.is<Solid::StorageDrive>()) {
|
||
|
Solid::StorageDrive *drive = device.as<Solid::StorageDrive>();
|
||
|
if (!drive->isHotpluggable()) {
|
||
|
#ifdef HOTPLUGENGINE_TIMING
|
||
|
qDebug() << "storage, but not pluggable, returning" << t.restart();
|
||
|
#endif
|
||
|
return;
|
||
|
}
|
||
|
} else if (device.is<Solid::StorageVolume>()) {
|
||
|
Solid::StorageVolume *volume = device.as<Solid::StorageVolume>();
|
||
|
Solid::StorageVolume::UsageType type = volume->usage();
|
||
|
if ((type == Solid::StorageVolume::Unused || type == Solid::StorageVolume::PartitionTable) && !device.is<Solid::OpticalDisc>()) {
|
||
|
#ifdef HOTPLUGENGINE_TIMING
|
||
|
qDebug() << "storage volume, but not of interest" << t.restart();
|
||
|
#endif
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
m_devices.insert(device.udi(), device);
|
||
|
|
||
|
if (m_predicates.isEmpty()) {
|
||
|
findPredicates();
|
||
|
}
|
||
|
|
||
|
const QStringList interestingDesktopFiles = predicatesForDevice(device);
|
||
|
const bool isEncryptedContainer = m_encryptedPredicate.matches(device);
|
||
|
|
||
|
if (!interestingDesktopFiles.isEmpty() || isEncryptedContainer) {
|
||
|
// qDebug() << device.product();
|
||
|
// qDebug() << device.vendor();
|
||
|
// qDebug() << "number of interesting desktop file : " << interestingDesktopFiles.size();
|
||
|
Plasma::DataEngine::Data data;
|
||
|
data.insert(QStringLiteral("added"), added);
|
||
|
data.insert(QStringLiteral("udi"), device.udi());
|
||
|
|
||
|
if (!device.description().isEmpty()) {
|
||
|
data.insert(QStringLiteral("text"), device.description());
|
||
|
} else {
|
||
|
data.insert(QStringLiteral("text"), QString(device.vendor() + QLatin1Char(' ') + device.product()));
|
||
|
}
|
||
|
data.insert(QStringLiteral("icon"), device.icon());
|
||
|
data.insert(QStringLiteral("emblems"), device.emblems());
|
||
|
data.insert(QStringLiteral("predicateFiles"), interestingDesktopFiles);
|
||
|
data.insert(QStringLiteral("actions"), actionsForPredicates(interestingDesktopFiles));
|
||
|
|
||
|
data.insert(QStringLiteral("isEncryptedContainer"), isEncryptedContainer);
|
||
|
|
||
|
setData(device.udi(), data);
|
||
|
// qDebug() << "add hardware solid : " << udi;
|
||
|
}
|
||
|
|
||
|
#ifdef HOTPLUGENGINE_TIMING
|
||
|
qDebug() << "total time" << t.restart();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void HotplugEngine::onDeviceRemoved(const QString &udi)
|
||
|
{
|
||
|
// qDebug() << "remove hardware:" << udi;
|
||
|
|
||
|
if (m_startList.contains(udi)) {
|
||
|
m_startList.remove(udi);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
m_devices.remove(udi);
|
||
|
removeSource(udi);
|
||
|
}
|
||
|
|
||
|
K_PLUGIN_CLASS_WITH_JSON(HotplugEngine, "plasma-dataengine-hotplug.json")
|
||
|
|
||
|
#include "hotplugengine.moc"
|