2008-11-03 23:08:39 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2006-2007 Aaron Seigo <aseigo@kde.org>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Library General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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 General Public License for more details
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this program; if not, write to the
|
|
|
|
* Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "dataengine.h"
|
|
|
|
#include "private/dataengine_p.h"
|
2010-10-01 16:58:15 +00:00
|
|
|
#include "private/datacontainer_p.h"
|
2008-11-03 23:08:39 +00:00
|
|
|
|
2013-12-30 10:21:36 +01:00
|
|
|
#include <QAbstractItemModel>
|
2008-11-03 23:08:39 +00:00
|
|
|
#include <QQueue>
|
|
|
|
#include <QTimer>
|
|
|
|
#include <QTime>
|
|
|
|
#include <QTimerEvent>
|
|
|
|
#include <QVariant>
|
|
|
|
|
2013-07-29 19:05:59 +02:00
|
|
|
#include <QDebug>
|
2014-04-26 14:29:54 +02:00
|
|
|
#include <QStandardPaths>
|
|
|
|
|
2008-11-04 02:04:34 +00:00
|
|
|
#include <kplugininfo.h>
|
|
|
|
#include <kservice.h>
|
2012-11-18 20:26:13 +01:00
|
|
|
#include <klocalizedstring.h>
|
2008-11-03 23:08:39 +00:00
|
|
|
|
|
|
|
#include "datacontainer.h"
|
|
|
|
#include "package.h"
|
2011-07-19 22:04:22 +02:00
|
|
|
#include "pluginloader.h"
|
2008-11-03 23:08:39 +00:00
|
|
|
#include "service.h"
|
|
|
|
#include "scripting/dataenginescript.h"
|
|
|
|
|
2011-06-03 16:04:50 +02:00
|
|
|
#include "private/datacontainer_p.h"
|
2008-11-03 23:08:39 +00:00
|
|
|
#include "private/service_p.h"
|
2010-07-12 20:51:12 +00:00
|
|
|
#include "private/storage_p.h"
|
2014-04-27 17:14:22 +02:00
|
|
|
#include "config-plasma.h"
|
2008-11-03 23:08:39 +00:00
|
|
|
|
|
|
|
namespace Plasma
|
|
|
|
{
|
|
|
|
|
2013-07-28 13:06:32 +02:00
|
|
|
DataEngine::DataEngine(const KPluginInfo &plugin, QObject *parent)
|
2013-04-04 17:43:30 +02:00
|
|
|
: QObject(parent),
|
2013-07-28 13:06:32 +02:00
|
|
|
d(new DataEnginePrivate(this, plugin))
|
2013-04-04 17:43:30 +02:00
|
|
|
{
|
2013-04-04 18:07:18 +02:00
|
|
|
if (d->script) {
|
|
|
|
d->setupScriptSupport();
|
|
|
|
d->script->init();
|
|
|
|
} else {
|
|
|
|
// default implementation does nothing. this is for engines that have to
|
|
|
|
// start things in motion external to themselves before they can work
|
|
|
|
}
|
2013-04-04 17:43:30 +02:00
|
|
|
}
|
|
|
|
|
2014-04-26 01:45:47 +02:00
|
|
|
DataEngine::DataEngine(QObject *parent, const QVariantList &args)
|
2013-08-06 15:06:07 +02:00
|
|
|
: QObject(parent),
|
2014-05-06 20:52:42 +02:00
|
|
|
d(new DataEnginePrivate(this, KPluginInfo(args), args))
|
2013-06-28 05:09:27 +02:00
|
|
|
{
|
2013-07-04 02:58:26 +02:00
|
|
|
if (d->script) {
|
|
|
|
d->setupScriptSupport();
|
|
|
|
d->script->init();
|
|
|
|
}
|
2013-06-28 05:09:27 +02:00
|
|
|
}
|
|
|
|
|
2008-11-03 23:08:39 +00:00
|
|
|
DataEngine::~DataEngine()
|
|
|
|
{
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << objectName() << ": bye bye birdy! ";
|
2008-11-03 23:08:39 +00:00
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList DataEngine::sources() const
|
|
|
|
{
|
2008-11-08 20:44:03 +00:00
|
|
|
if (d->script) {
|
|
|
|
return d->script->sources();
|
|
|
|
} else {
|
|
|
|
return d->sources.keys();
|
|
|
|
}
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Service *DataEngine::serviceForSource(const QString &source)
|
|
|
|
{
|
2008-11-08 20:44:03 +00:00
|
|
|
if (d->script) {
|
2014-04-26 01:45:47 +02:00
|
|
|
Service *s = d->script->serviceForSource(source);
|
2010-03-10 21:41:37 +00:00
|
|
|
if (s) {
|
|
|
|
return s;
|
|
|
|
}
|
2008-11-08 20:44:03 +00:00
|
|
|
}
|
2010-03-10 21:41:37 +00:00
|
|
|
|
|
|
|
return new NullService(source, this);
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
2013-04-03 20:41:59 +02:00
|
|
|
KPluginInfo DataEngine::pluginInfo() const
|
|
|
|
{
|
|
|
|
return d->dataEngineDescription;
|
|
|
|
}
|
|
|
|
|
2008-11-03 23:08:39 +00:00
|
|
|
void DataEngine::connectSource(const QString &source, QObject *visualization,
|
|
|
|
uint pollingInterval,
|
2013-05-14 20:27:32 +02:00
|
|
|
Plasma::Types::IntervalAlignment intervalAlignment) const
|
2008-11-03 23:08:39 +00:00
|
|
|
{
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "connectSource" << source;
|
2008-11-03 23:08:39 +00:00
|
|
|
bool newSource;
|
|
|
|
DataContainer *s = d->requestSource(source, &newSource);
|
|
|
|
|
|
|
|
if (s) {
|
|
|
|
// we suppress the immediate invocation of dataUpdated here if the
|
|
|
|
// source was prexisting and they don't request delayed updates
|
|
|
|
// (we want to do an immediate update in that case so they don't
|
|
|
|
// have to wait for the first time out)
|
2010-04-20 19:50:41 +00:00
|
|
|
if (newSource && !s->data().isEmpty()) {
|
|
|
|
newSource = false;
|
|
|
|
}
|
2008-11-03 23:08:39 +00:00
|
|
|
d->connectSource(s, visualization, pollingInterval, intervalAlignment,
|
|
|
|
!newSource || pollingInterval > 0);
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << " ==> source connected";
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::connectAllSources(QObject *visualization, uint pollingInterval,
|
2013-05-14 20:27:32 +02:00
|
|
|
Plasma::Types::IntervalAlignment intervalAlignment) const
|
2008-11-03 23:08:39 +00:00
|
|
|
{
|
|
|
|
foreach (DataContainer *s, d->sources) {
|
|
|
|
d->connectSource(s, visualization, pollingInterval, intervalAlignment);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::disconnectSource(const QString &source, QObject *visualization) const
|
|
|
|
{
|
|
|
|
DataContainer *s = d->source(source, false);
|
|
|
|
|
|
|
|
if (s) {
|
|
|
|
s->disconnectVisualization(visualization);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DataContainer *DataEngine::containerForSource(const QString &source)
|
|
|
|
{
|
|
|
|
return d->source(source, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DataEngine::sourceRequestEvent(const QString &name)
|
|
|
|
{
|
|
|
|
if (d->script) {
|
|
|
|
return d->script->sourceRequestEvent(name);
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DataEngine::updateSourceEvent(const QString &source)
|
|
|
|
{
|
|
|
|
if (d->script) {
|
|
|
|
return d->script->updateSourceEvent(source);
|
|
|
|
} else {
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << source;
|
2008-11-03 23:08:39 +00:00
|
|
|
return false; //TODO: should this be true to trigger, even needless, updates on every tick?
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::setData(const QString &source, const QVariant &value)
|
|
|
|
{
|
|
|
|
setData(source, source, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::setData(const QString &source, const QString &key, const QVariant &value)
|
|
|
|
{
|
|
|
|
DataContainer *s = d->source(source, false);
|
|
|
|
bool isNew = !s;
|
|
|
|
|
|
|
|
if (isNew) {
|
|
|
|
s = d->source(source);
|
|
|
|
}
|
|
|
|
|
|
|
|
s->setData(key, value);
|
|
|
|
|
2011-09-01 19:07:22 +02:00
|
|
|
if (isNew && source != d->waitingSourceRequest) {
|
2008-11-03 23:08:39 +00:00
|
|
|
emit sourceAdded(source);
|
|
|
|
}
|
|
|
|
|
2013-04-04 16:46:41 +02:00
|
|
|
d->scheduleSourcesUpdated();
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
2013-05-02 18:02:38 +02:00
|
|
|
void DataEngine::setData(const QString &source, const QVariantMap &data)
|
2008-11-03 23:08:39 +00:00
|
|
|
{
|
|
|
|
DataContainer *s = d->source(source, false);
|
|
|
|
bool isNew = !s;
|
|
|
|
|
|
|
|
if (isNew) {
|
|
|
|
s = d->source(source);
|
|
|
|
}
|
|
|
|
|
|
|
|
Data::const_iterator it = data.constBegin();
|
|
|
|
while (it != data.constEnd()) {
|
|
|
|
s->setData(it.key(), it.value());
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
2011-09-01 19:07:22 +02:00
|
|
|
if (isNew && source != d->waitingSourceRequest) {
|
2008-11-03 23:08:39 +00:00
|
|
|
emit sourceAdded(source);
|
|
|
|
}
|
|
|
|
|
2013-04-04 16:46:41 +02:00
|
|
|
d->scheduleSourcesUpdated();
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::removeAllData(const QString &source)
|
|
|
|
{
|
|
|
|
DataContainer *s = d->source(source, false);
|
|
|
|
if (s) {
|
|
|
|
s->removeAllData();
|
2013-04-04 16:46:41 +02:00
|
|
|
d->scheduleSourcesUpdated();
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::removeData(const QString &source, const QString &key)
|
|
|
|
{
|
|
|
|
DataContainer *s = d->source(source, false);
|
|
|
|
if (s) {
|
|
|
|
s->setData(key, QVariant());
|
2013-04-04 16:46:41 +02:00
|
|
|
d->scheduleSourcesUpdated();
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-30 10:21:36 +01:00
|
|
|
void DataEngine::setModel(const QString &source, QAbstractItemModel *model)
|
|
|
|
{
|
|
|
|
if (model) {
|
2016-02-29 00:08:05 +01:00
|
|
|
setData(source, QStringLiteral("HasModel"), true);
|
2013-12-30 10:21:36 +01:00
|
|
|
} else {
|
2016-02-29 00:08:05 +01:00
|
|
|
removeData(source, QStringLiteral("HasModel"));
|
2013-12-30 10:21:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Plasma::DataContainer *s = containerForSource(source);
|
|
|
|
|
|
|
|
if (s) {
|
|
|
|
s->setModel(model);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-30 10:41:14 +01:00
|
|
|
QAbstractItemModel *DataEngine::modelForSource(const QString &source)
|
2013-12-30 10:21:36 +01:00
|
|
|
{
|
|
|
|
Plasma::DataContainer *s = containerForSource(source);
|
|
|
|
|
|
|
|
if (s) {
|
|
|
|
return s->model();
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-03 23:08:39 +00:00
|
|
|
void DataEngine::addSource(DataContainer *source)
|
|
|
|
{
|
|
|
|
if (d->sources.contains(source->objectName())) {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2015-12-15 16:56:40 -05:00
|
|
|
// qCDebug(LOG_PLASMA) << "source named \"" << source->objectName() << "\" already exists.";
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2008-11-03 23:08:39 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
QObject::connect(source, SIGNAL(updateRequested(DataContainer*)),
|
|
|
|
this, SLOT(internalUpdateSource(DataContainer*)));
|
2010-06-10 20:38:36 +00:00
|
|
|
QObject::connect(source, SIGNAL(destroyed(QObject*)), this, SLOT(sourceDestroyed(QObject*)));
|
2008-11-03 23:08:39 +00:00
|
|
|
d->sources.insert(source->objectName(), source);
|
|
|
|
emit sourceAdded(source->objectName());
|
2013-04-04 16:46:41 +02:00
|
|
|
d->scheduleSourcesUpdated();
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::setMinimumPollingInterval(int minimumMs)
|
|
|
|
{
|
|
|
|
d->minPollingInterval = minimumMs;
|
|
|
|
}
|
|
|
|
|
|
|
|
int DataEngine::minimumPollingInterval() const
|
|
|
|
{
|
|
|
|
return d->minPollingInterval;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::setPollingInterval(uint frequency)
|
|
|
|
{
|
|
|
|
killTimer(d->updateTimerId);
|
|
|
|
d->updateTimerId = 0;
|
|
|
|
|
|
|
|
if (frequency > 0) {
|
|
|
|
d->updateTimerId = startTimer(frequency);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::removeSource(const QString &source)
|
|
|
|
{
|
2014-04-26 01:45:47 +02:00
|
|
|
QHash<QString, DataContainer *>::iterator it = d->sources.find(source);
|
2008-11-03 23:08:39 +00:00
|
|
|
if (it != d->sources.end()) {
|
|
|
|
DataContainer *s = it.value();
|
2010-10-01 16:58:15 +00:00
|
|
|
s->d->store();
|
2016-05-19 22:45:29 +03:00
|
|
|
d->sources.erase(it);
|
2010-06-09 19:09:13 +00:00
|
|
|
s->disconnect(this);
|
2008-11-03 23:08:39 +00:00
|
|
|
s->deleteLater();
|
|
|
|
emit sourceRemoved(source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::removeAllSources()
|
|
|
|
{
|
2014-04-26 01:45:47 +02:00
|
|
|
QMutableHashIterator<QString, Plasma::DataContainer *> it(d->sources);
|
2008-11-03 23:08:39 +00:00
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
2010-05-04 03:28:09 +00:00
|
|
|
Plasma::DataContainer *s = it.value();
|
2011-11-30 15:19:58 +01:00
|
|
|
const QString source = it.key();
|
2011-11-30 09:17:01 +01:00
|
|
|
it.remove();
|
2011-10-03 14:06:34 +02:00
|
|
|
s->disconnect(this);
|
2011-11-28 10:37:21 +01:00
|
|
|
s->deleteLater();
|
2011-11-30 09:17:01 +01:00
|
|
|
emit sourceRemoved(source);
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DataEngine::isValid() const
|
|
|
|
{
|
|
|
|
return d->valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DataEngine::isEmpty() const
|
|
|
|
{
|
|
|
|
return d->sources.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::setValid(bool valid)
|
|
|
|
{
|
|
|
|
d->valid = valid;
|
|
|
|
}
|
|
|
|
|
2014-04-26 01:45:47 +02:00
|
|
|
QHash<QString, DataContainer *> DataEngine::containerDict() const
|
2008-11-03 23:08:39 +00:00
|
|
|
{
|
|
|
|
return d->sources;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::timerEvent(QTimerEvent *event)
|
|
|
|
{
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA);
|
2009-05-09 00:32:45 +00:00
|
|
|
if (event->timerId() == d->updateTimerId) {
|
|
|
|
// if the freq update is less than 0, don't bother
|
|
|
|
if (d->minPollingInterval < 0) {
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "uh oh.. no polling allowed!";
|
2009-05-09 00:32:45 +00:00
|
|
|
return;
|
|
|
|
}
|
2008-11-03 23:08:39 +00:00
|
|
|
|
2009-05-09 00:32:45 +00:00
|
|
|
// minPollingInterval
|
2014-06-23 13:03:16 +02:00
|
|
|
if (d->updateTimer.elapsed() < d->minPollingInterval) {
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "hey now.. slow down!";
|
2009-05-09 00:32:45 +00:00
|
|
|
return;
|
|
|
|
}
|
2008-11-03 23:08:39 +00:00
|
|
|
|
2014-06-23 13:03:16 +02:00
|
|
|
d->updateTimer.start();
|
2009-05-09 00:32:45 +00:00
|
|
|
updateAllSources();
|
|
|
|
} else if (event->timerId() == d->checkSourcesTimerId) {
|
2009-08-02 21:56:51 +00:00
|
|
|
killTimer(d->checkSourcesTimerId);
|
|
|
|
d->checkSourcesTimerId = 0;
|
|
|
|
|
2014-04-26 01:45:47 +02:00
|
|
|
QHashIterator<QString, Plasma::DataContainer *> it(d->sources);
|
2009-05-09 00:32:45 +00:00
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
|
|
|
it.value()->checkForUpdate();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
QObject::timerEvent(event);
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
2008-11-13 07:40:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DataEngine::updateAllSources()
|
|
|
|
{
|
2014-04-26 01:45:47 +02:00
|
|
|
QHashIterator<QString, Plasma::DataContainer *> it(d->sources);
|
2008-11-03 23:08:39 +00:00
|
|
|
while (it.hasNext()) {
|
|
|
|
it.next();
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "updating" << it.key();
|
2014-06-11 17:17:04 +02:00
|
|
|
if (it.value()->isUsed()) {
|
|
|
|
updateSourceEvent(it.key());
|
|
|
|
}
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
2013-04-04 16:46:41 +02:00
|
|
|
d->scheduleSourcesUpdated();
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
2009-06-27 01:26:33 +00:00
|
|
|
void DataEngine::forceImmediateUpdateOfAllVisualizations()
|
|
|
|
{
|
|
|
|
foreach (DataContainer *source, d->sources) {
|
2014-06-11 17:17:04 +02:00
|
|
|
if (source->isUsed()) {
|
|
|
|
source->forceImmediateUpdate();
|
|
|
|
}
|
2009-06-27 01:26:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-15 13:01:17 +02:00
|
|
|
Package DataEngine::package() const
|
2008-11-03 23:08:39 +00:00
|
|
|
{
|
2011-07-15 13:01:17 +02:00
|
|
|
return d->package ? *d->package : Package();
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
2010-08-25 21:35:40 +00:00
|
|
|
void DataEngine::setStorageEnabled(const QString &source, bool store)
|
2010-07-12 20:51:12 +00:00
|
|
|
{
|
|
|
|
DataContainer *s = d->source(source, false);
|
2010-10-01 16:58:15 +00:00
|
|
|
if (s) {
|
|
|
|
s->setStorageEnabled(store);
|
|
|
|
}
|
2010-07-12 20:51:12 +00:00
|
|
|
}
|
|
|
|
|
2008-11-03 23:08:39 +00:00
|
|
|
// Private class implementations
|
2014-05-06 20:52:42 +02:00
|
|
|
DataEnginePrivate::DataEnginePrivate(DataEngine *e, const KPluginInfo &info, const QVariantList &args)
|
2008-11-03 23:08:39 +00:00
|
|
|
: q(e),
|
2010-08-06 01:26:11 +00:00
|
|
|
dataEngineDescription(info),
|
2008-11-03 23:08:39 +00:00
|
|
|
refCount(-1), // first ref
|
2009-05-09 00:32:45 +00:00
|
|
|
checkSourcesTimerId(0),
|
2008-11-03 23:08:39 +00:00
|
|
|
updateTimerId(0),
|
|
|
|
minPollingInterval(-1),
|
2016-05-19 22:45:29 +03:00
|
|
|
valid(false),
|
2008-11-03 23:08:39 +00:00
|
|
|
script(0),
|
2013-02-11 21:38:18 +01:00
|
|
|
package(0)
|
2008-11-03 23:08:39 +00:00
|
|
|
{
|
2014-06-23 13:03:16 +02:00
|
|
|
updateTimer.start();
|
2008-11-03 23:08:39 +00:00
|
|
|
|
2013-07-03 05:13:25 +02:00
|
|
|
if (dataEngineDescription.isValid()) {
|
2016-05-19 22:45:29 +03:00
|
|
|
valid = true;
|
2013-07-03 05:13:25 +02:00
|
|
|
e->setObjectName(dataEngineDescription.name());
|
2013-04-10 01:50:31 +02:00
|
|
|
}
|
2008-11-03 23:08:39 +00:00
|
|
|
|
|
|
|
if (dataEngineDescription.isValid()) {
|
2016-02-29 00:08:05 +01:00
|
|
|
QString api = dataEngineDescription.property(QStringLiteral("X-Plasma-API")).toString();
|
2008-11-03 23:08:39 +00:00
|
|
|
|
|
|
|
if (!api.isEmpty()) {
|
|
|
|
const QString path =
|
2013-02-08 11:41:31 +01:00
|
|
|
QStandardPaths::locate(QStandardPaths::GenericDataLocation,
|
2014-04-27 17:14:22 +02:00
|
|
|
PLASMA_RELATIVE_DATA_INSTALL_DIR "/dataengines/" + dataEngineDescription.pluginName() + '/',
|
2013-02-08 11:41:31 +01:00
|
|
|
QStandardPaths::LocateDirectory);
|
2016-02-29 00:08:05 +01:00
|
|
|
package = new Package(PluginLoader::self()->loadPackage(QStringLiteral("Plasma/DataEngine"), api));
|
2011-07-15 13:01:17 +02:00
|
|
|
package->setPath(path);
|
|
|
|
|
|
|
|
if (package->isValid()) {
|
2014-05-06 20:52:42 +02:00
|
|
|
script = Plasma::loadScriptEngine(api, q, args);
|
2011-07-15 13:01:17 +02:00
|
|
|
}
|
2008-11-03 23:08:39 +00:00
|
|
|
|
|
|
|
if (!script) {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2015-12-15 16:56:40 -05:00
|
|
|
// qCDebug(LOG_PLASMA) << "Could not create a" << api << "ScriptEngine for the"
|
2013-08-01 12:15:38 +02:00
|
|
|
// << dataEngineDescription.name() << "DataEngine.";
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2008-11-03 23:08:39 +00:00
|
|
|
delete package;
|
|
|
|
package = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DataEnginePrivate::~DataEnginePrivate()
|
|
|
|
{
|
|
|
|
delete script;
|
|
|
|
script = 0;
|
|
|
|
delete package;
|
|
|
|
package = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEnginePrivate::internalUpdateSource(DataContainer *source)
|
|
|
|
{
|
|
|
|
if (minPollingInterval > 0 &&
|
2014-04-26 01:45:47 +02:00
|
|
|
source->timeSinceLastUpdate() < (uint)minPollingInterval) {
|
2008-11-03 23:08:39 +00:00
|
|
|
// skip updating this source; it's been too soon
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "internal update source is delaying" << source->timeSinceLastUpdate() << minPollingInterval;
|
2008-11-03 23:08:39 +00:00
|
|
|
//but fake an update so that the signalrelay that triggered this gets the data from the
|
|
|
|
//recent update. this way we don't have to worry about queuing - the relay will send a
|
|
|
|
//signal immediately and everyone else is undisturbed.
|
|
|
|
source->setNeedsUpdate();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (q->updateSourceEvent(source->objectName())) {
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "queuing an update";
|
2013-04-04 16:46:41 +02:00
|
|
|
scheduleSourcesUpdated();
|
2008-11-03 23:08:39 +00:00
|
|
|
}/* else {
|
2011-07-29 15:46:52 +02:00
|
|
|
#ifndef NDEBUG
|
2015-12-15 16:56:40 -05:00
|
|
|
// qCDebug(LOG_PLASMA) << "no update";
|
2011-07-29 15:46:52 +02:00
|
|
|
#endif
|
2008-11-03 23:08:39 +00:00
|
|
|
}*/
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEnginePrivate::ref()
|
|
|
|
{
|
|
|
|
--refCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEnginePrivate::deref()
|
|
|
|
{
|
|
|
|
++refCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DataEnginePrivate::isUsed() const
|
|
|
|
{
|
|
|
|
return refCount != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
DataContainer *DataEnginePrivate::source(const QString &sourceName, bool createWhenMissing)
|
|
|
|
{
|
2014-04-26 01:45:47 +02:00
|
|
|
QHash<QString, DataContainer *>::const_iterator it = sources.constFind(sourceName);
|
2008-11-03 23:08:39 +00:00
|
|
|
if (it != sources.constEnd()) {
|
|
|
|
DataContainer *s = it.value();
|
2010-08-25 21:17:54 +00:00
|
|
|
return s;
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!createWhenMissing) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "DataEngine " << q->objectName() << ": could not find DataContainer " << sourceName << ", creating";
|
2008-11-03 23:08:39 +00:00
|
|
|
DataContainer *s = new DataContainer(q);
|
|
|
|
s->setObjectName(sourceName);
|
|
|
|
sources.insert(sourceName, s);
|
2012-01-21 21:11:26 -05:00
|
|
|
QObject::connect(s, SIGNAL(destroyed(QObject*)), q, SLOT(sourceDestroyed(QObject*)));
|
2008-11-03 23:08:39 +00:00
|
|
|
QObject::connect(s, SIGNAL(updateRequested(DataContainer*)),
|
|
|
|
q, SLOT(internalUpdateSource(DataContainer*)));
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataEnginePrivate::connectSource(DataContainer *s, QObject *visualization,
|
|
|
|
uint pollingInterval,
|
2013-05-14 20:27:32 +02:00
|
|
|
Plasma::Types::IntervalAlignment align,
|
2008-11-03 23:08:39 +00:00
|
|
|
bool immediateCall)
|
|
|
|
{
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "connect source called" << s->objectName() << "with interval" << pollingInterval;
|
2010-02-18 10:05:31 +00:00
|
|
|
|
2008-11-03 23:08:39 +00:00
|
|
|
if (pollingInterval > 0) {
|
|
|
|
// never more frequently than allowed, never more than 20 times per second
|
|
|
|
uint min = qMax(50, minPollingInterval); // for qMax below
|
|
|
|
pollingInterval = qMax(min, pollingInterval);
|
|
|
|
|
|
|
|
// align on the 50ms
|
|
|
|
pollingInterval = pollingInterval - (pollingInterval % 50);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (immediateCall) {
|
|
|
|
// we don't want to do an immediate call if we are simply
|
|
|
|
// reconnecting
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "immediate call requested, we have:" << s->visualizationIsConnected(visualization);
|
2009-08-20 09:34:36 +00:00
|
|
|
immediateCall = !s->data().isEmpty() &&
|
|
|
|
!s->visualizationIsConnected(visualization);
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
s->connectVisualization(visualization, pollingInterval, align);
|
|
|
|
|
|
|
|
if (immediateCall) {
|
|
|
|
QMetaObject::invokeMethod(visualization, "dataUpdated",
|
|
|
|
Q_ARG(QString, s->objectName()),
|
|
|
|
Q_ARG(Plasma::DataEngine::Data, s->data()));
|
2013-12-24 15:21:21 +01:00
|
|
|
if (s->d->model) {
|
|
|
|
QMetaObject::invokeMethod(visualization, "modelChanged",
|
2014-04-26 01:45:47 +02:00
|
|
|
Q_ARG(QString, s->objectName()),
|
|
|
|
Q_ARG(QAbstractItemModel *, s->d->model.data()));
|
2013-12-24 15:21:21 +01:00
|
|
|
}
|
2011-03-02 22:04:53 +01:00
|
|
|
s->d->dirty = false;
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-09 18:04:28 +00:00
|
|
|
void DataEnginePrivate::sourceDestroyed(QObject *object)
|
|
|
|
{
|
2014-04-26 01:45:47 +02:00
|
|
|
QHash<QString, DataContainer *>::iterator it = sources.begin();
|
2010-06-09 19:09:13 +00:00
|
|
|
while (it != sources.end()) {
|
2010-06-09 18:04:28 +00:00
|
|
|
if (it.value() == object) {
|
2010-06-09 19:09:13 +00:00
|
|
|
sources.erase(it);
|
2011-11-28 10:42:40 +01:00
|
|
|
emit q->sourceRemoved(object->objectName());
|
2010-06-09 18:04:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-03 23:08:39 +00:00
|
|
|
DataContainer *DataEnginePrivate::requestSource(const QString &sourceName, bool *newSource)
|
|
|
|
{
|
|
|
|
if (newSource) {
|
|
|
|
*newSource = false;
|
|
|
|
}
|
|
|
|
|
2015-12-15 16:56:40 -05:00
|
|
|
//qCDebug(LOG_PLASMA) << "requesting source " << sourceName;
|
2008-11-03 23:08:39 +00:00
|
|
|
DataContainer *s = source(sourceName, false);
|
|
|
|
|
|
|
|
if (!s) {
|
|
|
|
// we didn't find a data source, so give the engine an opportunity to make one
|
2015-12-15 16:56:40 -05:00
|
|
|
/*// qCDebug(LOG_PLASMA) << "DataEngine " << q->objectName()
|
2008-11-03 23:08:39 +00:00
|
|
|
<< ": could not find DataContainer " << sourceName
|
|
|
|
<< " will create on request" << endl;*/
|
2011-09-01 19:07:22 +02:00
|
|
|
waitingSourceRequest = sourceName;
|
2008-11-03 23:08:39 +00:00
|
|
|
if (q->sourceRequestEvent(sourceName)) {
|
|
|
|
s = source(sourceName, false);
|
|
|
|
if (s) {
|
|
|
|
// now we have a source; since it was created on demand, assume
|
|
|
|
// it should be removed when not used
|
|
|
|
if (newSource) {
|
|
|
|
*newSource = true;
|
|
|
|
}
|
|
|
|
QObject::connect(s, SIGNAL(becameUnused(QString)), q, SLOT(removeSource(QString)));
|
2011-09-01 19:07:22 +02:00
|
|
|
emit q->sourceAdded(sourceName);
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
}
|
2011-09-01 19:07:22 +02:00
|
|
|
waitingSourceRequest.clear();
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2009-11-12 06:38:16 +00:00
|
|
|
// put all setup routines for script here. at this point we can assume that
|
|
|
|
// package exists and that we have a script engine
|
|
|
|
void DataEnginePrivate::setupScriptSupport()
|
|
|
|
{
|
|
|
|
if (!package) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-08-06 01:26:11 +00:00
|
|
|
/*
|
2014-04-26 01:45:47 +02:00
|
|
|
#ifndef NDEBUG
|
2015-12-15 16:56:40 -05:00
|
|
|
// qCDebug(LOG_PLASMA) << "sletting up script support, package is in" << package->path()
|
2014-04-26 01:45:47 +02:00
|
|
|
#endif
|
2009-11-12 06:38:16 +00:00
|
|
|
<< "which is a" << package->structure()->type() << "package"
|
|
|
|
<< ", main script is" << package->filePath("mainscript");
|
2010-08-06 01:26:11 +00:00
|
|
|
*/
|
2009-11-12 06:38:16 +00:00
|
|
|
|
2013-08-05 11:19:11 +02:00
|
|
|
// FIXME: Replace with ki18n functionality once semantics is clear.
|
|
|
|
// const QString translationsPath = package->filePath("translations");
|
|
|
|
// if (!translationsPath.isEmpty()) {
|
|
|
|
// KGlobal::dirs()->addResourceDir("locale", translationsPath);
|
|
|
|
// }
|
2009-11-12 06:38:16 +00:00
|
|
|
}
|
|
|
|
|
2013-04-04 16:46:41 +02:00
|
|
|
void DataEnginePrivate::scheduleSourcesUpdated()
|
|
|
|
{
|
|
|
|
if (checkSourcesTimerId) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
checkSourcesTimerId = q->startTimer(0);
|
|
|
|
}
|
|
|
|
|
2008-11-03 23:08:39 +00:00
|
|
|
}
|
|
|
|
|
2012-02-08 23:33:03 +01:00
|
|
|
#include "moc_dataengine.cpp"
|