You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
254 lines
7.9 KiB
254 lines
7.9 KiB
/*************************************************************************** |
|
* Copyright (C) 2007-2009 by Shawn Starr <shawn.starr@rogers.com> * |
|
* Copyright (C) 2009 by Aaron Seigo <aseigo@kde.org> * |
|
* * |
|
* This program is free software; you can redistribute it and/or modify * |
|
* it under the terms of the GNU General Public License as published by * |
|
* the Free Software Foundation; either version 2 of the License, 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 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 "weatherengine.h" |
|
|
|
#include <QTimer> |
|
#include <QDebug> |
|
|
|
#include <KSycoca> |
|
|
|
#include <Plasma/DataContainer> |
|
#include <Plasma/PluginLoader> |
|
|
|
#include "ions/ion.h" |
|
|
|
// Constructor |
|
WeatherEngine::WeatherEngine(QObject *parent, const QVariantList& args) |
|
: Plasma::DataEngine(parent, args), |
|
m_networkAvailable(false), |
|
m_networkAccessManager(new QNetworkAccessManager(this)) |
|
{ |
|
m_reconnectTimer.setSingleShot(true); |
|
connect(&m_reconnectTimer, &QTimer::timeout, this, &WeatherEngine::startReconnect); |
|
|
|
// Globally notify all plugins to remove their sources (and unload plugin) |
|
connect(this, &Plasma::DataEngine::sourceRemoved, this, &WeatherEngine::removeIonSource); |
|
|
|
// Get the list of available plugins but don't load them |
|
QNetworkAccessManager::NetworkAccessibility status = m_networkAccessManager->networkAccessible(); |
|
m_networkAvailable = (status == QNetworkAccessManager::Accessible || |
|
status == QNetworkAccessManager::UnknownAccessibility); |
|
connect(m_networkAccessManager, &QNetworkAccessManager::networkAccessibleChanged, |
|
this, &WeatherEngine::networkStatusChanged); |
|
|
|
connect(KSycoca::self(), static_cast<void (KSycoca::*)(const QStringList&)>(&KSycoca::databaseChanged), |
|
this, &WeatherEngine::updateIonList); |
|
|
|
updateIonList(); |
|
} |
|
|
|
// Destructor |
|
WeatherEngine::~WeatherEngine() |
|
{ |
|
} |
|
|
|
/** |
|
* Loads an ion plugin given a plugin name found via KService. |
|
*/ |
|
IonInterface *WeatherEngine::loadIon(const QString& plugName) |
|
{ |
|
KPluginInfo foundPlugin; |
|
|
|
foreach(const KPluginInfo &info, Plasma::PluginLoader::self()->listEngineInfo(QLatin1String("weatherengine"))) { |
|
if (info.pluginName() == plugName) { |
|
foundPlugin = info; |
|
break; |
|
} |
|
} |
|
|
|
if (!foundPlugin.isValid()) { |
|
return nullptr; |
|
} |
|
|
|
// Load the Ion plugin, store it into a QMap to handle multiple ions. |
|
Plasma::DataEngine *engine = dataEngine(foundPlugin.pluginName()); |
|
IonInterface *ion = qobject_cast<IonInterface*>(engine); |
|
|
|
if (!ion) { |
|
return nullptr; |
|
} |
|
|
|
ion->setObjectName(plugName); |
|
connect(ion, &DataEngine::sourceAdded, this, &WeatherEngine::newIonSource); |
|
connect(ion, &IonInterface::forceUpdate, this, &WeatherEngine::forceUpdate); |
|
|
|
m_ions << plugName; |
|
|
|
return ion; |
|
} |
|
|
|
/* FIXME: Q_PROPERTY functions to update the list of available plugins */ |
|
|
|
void WeatherEngine::updateIonList(const QStringList &changedResources) |
|
{ |
|
if (changedResources.isEmpty() || changedResources.contains(QStringLiteral("services"))) { |
|
removeAllData(QStringLiteral("ions")); |
|
foreach (const KPluginInfo &info, Plasma::PluginLoader::self()->listEngineInfo(QLatin1String("weatherengine"))) { |
|
const QString data = info.property(QStringLiteral("Name")).toString() + QLatin1Char('|') + info.pluginName(); |
|
setData(QStringLiteral("ions"), info.pluginName(), data); |
|
} |
|
} |
|
} |
|
|
|
/** |
|
* SLOT: Get data from a new source |
|
*/ |
|
void WeatherEngine::newIonSource(const QString& source) |
|
{ |
|
IonInterface *ion = qobject_cast<IonInterface*>(sender()); |
|
|
|
if (!ion) { |
|
return; |
|
} |
|
|
|
qDebug() << "newIonSource()"; |
|
ion->connectSource(source, this); |
|
} |
|
|
|
/** |
|
* SLOT: Remove the datasource from the ion and unload plugin if needed |
|
*/ |
|
void WeatherEngine::removeIonSource(const QString& source) |
|
{ |
|
IonInterface *ion = ionForSource(source); |
|
if (ion) { |
|
ion->removeSource(source); |
|
} |
|
qDebug() << "removeIonSource()"; |
|
} |
|
|
|
/** |
|
* SLOT: Push out new data to applet |
|
*/ |
|
void WeatherEngine::dataUpdated(const QString& source, const Plasma::DataEngine::Data& data) |
|
{ |
|
qDebug() << "dataUpdated()"; |
|
setData(source, data); |
|
} |
|
|
|
/** |
|
* SLOT: Set up each Ion for the first time and get any data |
|
*/ |
|
bool WeatherEngine::sourceRequestEvent(const QString &source) |
|
{ |
|
IonInterface* ion = ionForSource(source); |
|
|
|
if (!ion) { |
|
ion = loadIon(ionNameForSource(source)); |
|
if (!ion) { |
|
return false; |
|
} |
|
} |
|
|
|
// we should connect to the ion anyway, even if the network |
|
// is down. when it comes up again, then it will be refreshed |
|
ion->connectSource(source, this); |
|
|
|
qDebug() << "sourceRequestEvent(): Network is: " << m_networkAvailable; |
|
if (!m_networkAvailable) { |
|
setData(source, Data()); |
|
return true; |
|
} |
|
|
|
if (!containerForSource(source)) { |
|
// it is an async reply, we need to set up the data anyways |
|
setData(source, Data()); |
|
} |
|
return true; |
|
} |
|
|
|
/** |
|
* SLOT: update the Applet with new data from all ions loaded. |
|
*/ |
|
bool WeatherEngine::updateSourceEvent(const QString& source) |
|
{ |
|
IonInterface *ion = ionForSource(source); |
|
if (!ion) { |
|
return false; |
|
} |
|
|
|
qDebug() << "updateSourceEvent(): Network is: " << m_networkAvailable; |
|
if (!m_networkAvailable) { |
|
return false; |
|
} |
|
|
|
return ion->updateSourceEvent(source); |
|
} |
|
|
|
void WeatherEngine::networkStatusChanged(QNetworkAccessManager::NetworkAccessibility status) |
|
{ |
|
qDebug(); |
|
m_networkAvailable = status == QNetworkAccessManager::Accessible || status == QNetworkAccessManager::UnknownAccessibility; |
|
if (m_networkAvailable) { |
|
// allow the network to settle down and actually come up |
|
m_reconnectTimer.start(5000); |
|
} |
|
} |
|
|
|
void WeatherEngine::startReconnect() |
|
{ |
|
foreach (const QString &i, m_ions) { |
|
IonInterface * ion = qobject_cast<IonInterface *>(dataEngine(i)); |
|
qDebug() << "resetting" << ion; |
|
if (ion) { |
|
ion->reset(); |
|
} |
|
} |
|
} |
|
|
|
void WeatherEngine::forceUpdate(IonInterface *ion, const QString &source) |
|
{ |
|
Q_UNUSED(ion); |
|
Plasma::DataContainer *container = containerForSource(source); |
|
if (container) { |
|
qDebug() << "immediate update of" << source; |
|
container->forceImmediateUpdate(); |
|
} else { |
|
qDebug() << "innexplicable failure of" << source; |
|
} |
|
} |
|
|
|
IonInterface* WeatherEngine::ionForSource(const QString& name) |
|
{ |
|
const int offset = name.indexOf(QLatin1Char('|')); |
|
|
|
if (offset < 1) { |
|
return nullptr; |
|
} |
|
|
|
QString ionName = name.left(offset); |
|
return qobject_cast<IonInterface *>(dataEngine(ionName)); |
|
} |
|
|
|
QString WeatherEngine::ionNameForSource(const QString& source) const |
|
{ |
|
const int offset = source.indexOf(QLatin1Char('|')); |
|
|
|
if (offset < 1) { |
|
return QString(); |
|
} |
|
|
|
return source.left(offset); |
|
} |
|
|
|
K_EXPORT_PLASMA_DATAENGINE_WITH_JSON(weather, WeatherEngine, "plasma-dataengine-weather.json") |
|
|
|
#include "weatherengine.moc"
|
|
|