syncthingtray/fileitemactionplugin/syncthingfileitemaction.cpp

454 lines
17 KiB
C++
Raw Normal View History

2017-02-20 18:37:11 +01:00
#include "./syncthingfileitemaction.h"
#include "../model/syncthingicons.h"
#include "../connector/syncthingconfig.h"
#include "../connector/syncthingconnectionsettings.h"
#include "../connector/syncthingdir.h"
#include "../connector/utils.h"
#include <qtutilities/aboutdialog/aboutdialog.h>
2017-05-01 03:34:43 +02:00
#include <qtutilities/resources/resources.h>
2017-02-20 18:37:11 +01:00
#include <KFileItem>
#include <KPluginFactory>
#include <KPluginLoader>
#include <QAction>
#include <QDir>
#include <QEvent>
#include <QHBoxLayout>
#include <QLabel>
2017-05-01 03:34:43 +02:00
#include <QMenu>
#include <QMessageBox>
#include <QWidget>
2017-02-20 18:37:11 +01:00
#include <functional>
2017-05-01 03:34:43 +02:00
#include <iostream>
2017-02-20 18:37:11 +01:00
#include "resources/config.h"
using namespace std;
using namespace Dialogs;
using namespace Data;
K_PLUGIN_FACTORY(SyncthingFileItemActionFactory, registerPlugin<SyncthingFileItemAction>();)
2017-05-01 03:34:43 +02:00
struct SyncthingItem {
2017-02-20 18:37:11 +01:00
SyncthingItem(const SyncthingDir *dir, const QString &path);
const SyncthingDir *dir;
QString path;
QString name;
};
2017-05-01 03:34:43 +02:00
SyncthingItem::SyncthingItem(const SyncthingDir *dir, const QString &path)
: dir(dir)
, path(path)
2017-02-20 18:37:11 +01:00
{
int lastSep = path.lastIndexOf(QChar('/'));
2017-05-01 03:34:43 +02:00
if (lastSep > 0) {
2017-02-20 18:37:11 +01:00
name = path.mid(lastSep + 1);
} else {
name = path;
}
}
2017-05-01 03:34:43 +02:00
SyncthingMenuAction::SyncthingMenuAction(const KFileItemListProperties &properties, const QList<QAction *> &actions, QWidget *parentWidget)
: QAction(parentWidget)
, m_properties(properties)
{
2017-05-01 03:34:43 +02:00
if (!actions.isEmpty()) {
auto *menu = new QMenu(parentWidget);
menu->addActions(actions);
setMenu(menu);
}
updateStatus(SyncthingFileItemAction::connection().status());
}
void SyncthingMenuAction::updateStatus(SyncthingStatus status)
{
2017-05-01 03:34:43 +02:00
if (status != SyncthingStatus::Disconnected && status != SyncthingStatus::Reconnecting && status != SyncthingStatus::BeingDestroyed) {
setText(tr("Syncthing"));
setIcon(statusIcons().scanninig);
2017-05-01 03:34:43 +02:00
if (!menu()) {
const QList<QAction *> actions = SyncthingFileItemAction::createActions(m_properties, parentWidget());
2017-05-01 03:34:43 +02:00
if (!actions.isEmpty()) {
auto *menu = new QMenu(parentWidget());
menu->addActions(actions);
setMenu(menu);
}
}
} else {
2017-05-01 03:34:43 +02:00
if (status != SyncthingStatus::Reconnecting) {
SyncthingFileItemAction::connection().connect();
}
setText(tr("Syncthing - connecting"));
setIcon(statusIcons().disconnected);
2017-05-01 03:34:43 +02:00
if (QMenu *menu = this->menu()) {
setMenu(nullptr);
delete menu;
}
}
}
2017-05-01 03:34:43 +02:00
SyncthingInfoAction::SyncthingInfoAction(QObject *parent)
: QWidgetAction(parent)
{
}
QWidget *SyncthingInfoAction::createWidget(QWidget *parent)
{
auto *container = new QWidget(parent);
auto *layout = new QHBoxLayout(parent);
layout->setMargin(4);
layout->setSpacing(5);
auto *iconLabel = new QLabel(parent);
iconLabel->setPixmap(icon().pixmap(16));
iconLabel->setFixedWidth(16);
iconLabel->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Minimum);
layout->addWidget(iconLabel);
auto *textLabel = new QLabel(text(), parent);
layout->addWidget(textLabel);
container->setLayout(layout);
return container;
}
2017-05-01 03:34:43 +02:00
SyncthingDirActions::SyncthingDirActions(const SyncthingDir &dir, QObject *parent)
: QObject(parent)
, m_dirId(dir.id)
2017-03-13 00:40:48 +01:00
{
m_infoAction.setSeparator(true);
updateStatus(dir);
}
void SyncthingDirActions::updateStatus(const std::vector<SyncthingDir> &dirs)
{
2017-05-01 03:34:43 +02:00
for (const SyncthingDir &dir : dirs) {
if (updateStatus(dir)) {
2017-03-13 00:40:48 +01:00
return;
}
}
m_statusAction.setText(tr("Status: not available anymore"));
m_statusAction.setIcon(statusIcons().disconnected);
}
bool SyncthingDirActions::updateStatus(const SyncthingDir &dir)
{
2017-05-01 03:34:43 +02:00
if (dir.id != m_dirId) {
2017-03-13 00:40:48 +01:00
return false;
}
m_infoAction.setText(tr("Directory info for %1").arg(dir.displayName()));
m_infoAction.setIcon(QIcon::fromTheme(QStringLiteral("dialog-information")));
m_statusAction.setText(tr("Status: ") + dir.statusString());
2017-05-01 03:34:43 +02:00
if (dir.paused && dir.status != SyncthingDirStatus::OutOfSync) {
2017-03-13 00:40:48 +01:00
m_statusAction.setIcon(statusIcons().pause);
} else {
2017-05-01 03:34:43 +02:00
switch (dir.status) {
2017-03-13 00:40:48 +01:00
case SyncthingDirStatus::Unknown:
case SyncthingDirStatus::Unshared:
m_statusAction.setIcon(statusIcons().disconnected);
break;
case SyncthingDirStatus::Idle:
m_statusAction.setIcon(statusIcons().idling);
break;
case SyncthingDirStatus::Scanning:
m_statusAction.setIcon(statusIcons().scanninig);
break;
case SyncthingDirStatus::Synchronizing:
m_statusAction.setIcon(statusIcons().sync);
break;
case SyncthingDirStatus::OutOfSync:
m_statusAction.setIcon(statusIcons().error);
break;
}
}
m_globalStatusAction.setText(tr("Global: ") + directoryStatusString(dir.globalStats));
m_localStatusAction.setText(tr("Local: ") + directoryStatusString(dir.localStats));
2017-03-13 00:40:48 +01:00
m_lastScanAction.setText(tr("Last scan time: ") + agoString(dir.lastScanTime));
m_lastScanAction.setIcon(QIcon::fromTheme(QStringLiteral("accept_time_event")));
m_rescanIntervalAction.setText(tr("Rescan interval: %1 seconds").arg(dir.rescanInterval));
if (dir.itemErrors.empty()) {
m_errorsAction.setVisible(false);
} else {
m_errorsAction.setVisible(true);
m_errorsAction.setIcon(QIcon::fromTheme(QStringLiteral("dialog-error")));
m_errorsAction.setText(tr("%1 item(s) out-of-sync", nullptr, trQuandity(dir.itemErrors.size())).arg(dir.itemErrors.size()));
}
2017-03-13 00:40:48 +01:00
return true;
}
2017-05-01 03:34:43 +02:00
QList<QAction *> &operator<<(QList<QAction *> &actions, SyncthingDirActions &dirActions)
2017-03-13 00:40:48 +01:00
{
return actions << &dirActions.m_infoAction << &dirActions.m_statusAction << &dirActions.m_globalStatusAction << &dirActions.m_localStatusAction
<< &dirActions.m_lastScanAction << &dirActions.m_rescanIntervalAction << &dirActions.m_errorsAction;
2017-03-13 00:40:48 +01:00
}
2017-02-20 18:37:11 +01:00
SyncthingConnection SyncthingFileItemAction::s_connection;
2017-05-01 03:34:43 +02:00
SyncthingFileItemAction::SyncthingFileItemAction(QObject *parent, const QVariantList &)
: KAbstractFileItemActionPlugin(parent)
2017-02-20 18:37:11 +01:00
{
2017-05-01 03:34:43 +02:00
if (s_connection.apiKey().isEmpty()) {
2017-02-26 18:59:37 +01:00
// first initialization: load translations, determine config, establish connection
LOAD_QT_TRANSLATIONS;
2017-02-20 18:37:11 +01:00
// determine path of Syncthing config file
const QByteArray configPathFromEnv(qgetenv("KIO_SYNCTHING_CONFIG_PATH"));
2017-05-01 03:34:43 +02:00
const QString configPath = !configPathFromEnv.isEmpty() ? QString::fromLocal8Bit(configPathFromEnv) : SyncthingConfig::locateConfigFile();
if (configPath.isEmpty()) {
2017-02-20 18:37:11 +01:00
cerr << "Unable to determine location of Syncthing config. Set KIO_SYNCTHING_CONFIG_PATH to specify location." << endl;
return;
}
// load Syncthing config
SyncthingConfig config;
2017-05-01 03:34:43 +02:00
if (!config.restore(configPath)) {
2017-02-20 18:37:11 +01:00
cerr << "Unable to load Syncthing config from \"" << configPath.toLocal8Bit().data() << "\"" << endl;
2017-05-01 03:34:43 +02:00
if (configPathFromEnv.isEmpty()) {
2017-02-20 18:37:11 +01:00
cerr << "Note: Set KIO_SYNCTHING_CONFIG_PATH to specify config file explicitely." << endl;
}
return;
}
cerr << "Syncthing config loaded from \"" << configPath.toLocal8Bit().data() << "\"" << endl;
SyncthingConnectionSettings settings;
settings.syncthingUrl = config.syncthingUrl();
settings.apiKey.append(config.guiApiKey);
// establish connection
bool ok;
int reconnectInterval = qEnvironmentVariableIntValue("KIO_SYNCTHING_RECONNECT_INTERVAL", &ok);
2017-05-01 03:34:43 +02:00
if (!ok || reconnectInterval < 0) {
2017-02-20 18:37:11 +01:00
reconnectInterval = 10000;
}
s_connection.setAutoReconnectInterval(reconnectInterval);
s_connection.reconnect(settings);
connect(&s_connection, &SyncthingConnection::error, &SyncthingFileItemAction::logConnectionError);
connect(&s_connection, &SyncthingConnection::statusChanged, &SyncthingFileItemAction::logConnectionStatus);
}
}
QList<QAction *> SyncthingFileItemAction::actions(const KFileItemListProperties &fileItemInfo, QWidget *parentWidget)
{
// handle case when not connected yet
2017-05-01 03:34:43 +02:00
if (!s_connection.isConnected()) {
s_connection.connect();
auto *menuAction = new SyncthingMenuAction(fileItemInfo, QList<QAction *>(), parentWidget);
connect(&s_connection, &SyncthingConnection::statusChanged, menuAction, &SyncthingMenuAction::updateStatus);
return QList<QAction *>() << menuAction;
}
2017-03-17 00:34:53 +01:00
const QList<QAction *> actions = createActions(fileItemInfo, parentWidget);
// don't show anything if no relevant actions could be determined
2017-05-01 03:34:43 +02:00
if (actions.isEmpty()) {
return actions;
}
2017-05-01 03:34:43 +02:00
return QList<QAction *>() << new SyncthingMenuAction(fileItemInfo, actions, parentWidget);
}
SyncthingConnection &SyncthingFileItemAction::connection()
{
return s_connection;
}
void SyncthingFileItemAction::logConnectionStatus()
{
cerr << "Syncthing connection status changed to: " << s_connection.statusText().toLocal8Bit().data() << endl;
}
2017-03-27 11:12:06 +02:00
void SyncthingFileItemAction::logConnectionError(const QString &errorMessage, SyncthingErrorCategory errorCategory)
{
2017-05-01 03:34:43 +02:00
switch (errorCategory) {
2017-03-27 11:12:06 +02:00
case SyncthingErrorCategory::Parsing:
case SyncthingErrorCategory::SpecificRequest:
QMessageBox::critical(nullptr, tr("Syncthing connection error"), errorMessage);
break;
default:
cerr << "Syncthing connection error: " << errorMessage.toLocal8Bit().data() << endl;
}
}
void SyncthingFileItemAction::rescanDir(const QString &dirId, const QString &relpath)
{
s_connection.rescan(dirId, relpath);
}
void SyncthingFileItemAction::showAboutDialog()
{
2017-05-01 03:34:43 +02:00
auto *aboutDialog = new AboutDialog(nullptr, QStringLiteral(APP_NAME), QStringLiteral(APP_AUTHOR "\nSyncthing icons from Syncthing project"),
QStringLiteral(APP_VERSION), QStringLiteral(APP_URL), QStringLiteral(APP_DESCRIPTION), QImage(statusIcons().scanninig.pixmap(128).toImage()));
aboutDialog->setWindowTitle(tr("About") + QStringLiteral(" - " APP_NAME));
aboutDialog->setWindowIcon(QIcon::fromTheme(QStringLiteral("syncthingtray")));
aboutDialog->setAttribute(Qt::WA_DeleteOnClose);
aboutDialog->show();
}
QList<QAction *> SyncthingFileItemAction::createActions(const KFileItemListProperties &fileItemInfo, QWidget *parentWidget)
{
2017-05-01 03:34:43 +02:00
QList<QAction *> actions;
2017-02-20 18:37:11 +01:00
// check whether any directories are known
const auto &dirs = s_connection.dirInfo();
2017-05-01 03:34:43 +02:00
if (dirs.empty()) {
return actions;
2017-02-20 18:37:11 +01:00
}
// get all paths
QStringList paths;
paths.reserve(fileItemInfo.items().size());
2017-05-01 03:34:43 +02:00
for (const KFileItem &item : fileItemInfo.items()) {
if (!item.isLocalFile()) {
2017-02-20 18:37:11 +01:00
// don't show any actions when remote files are selected
2017-05-01 03:34:43 +02:00
return QList<QAction *>();
2017-02-20 18:37:11 +01:00
}
paths << item.localPath();
}
// determine relevant Syncthing dirs
QList<const SyncthingDir *> detectedDirs;
QList<const SyncthingDir *> containingDirs;
QList<SyncthingItem> detectedItems;
const SyncthingDir *lastDir;
2017-05-01 03:34:43 +02:00
for (const SyncthingDir &dir : dirs) {
QStringRef dirPath(dir.pathWithoutTrailingSlash());
2017-05-01 03:34:43 +02:00
for (const QString &path : paths) {
if (path == dirPath) {
2017-02-25 18:35:27 +01:00
lastDir = &dir;
2017-05-01 03:34:43 +02:00
if (!detectedDirs.contains(lastDir)) {
2017-02-25 18:35:27 +01:00
detectedDirs << lastDir;
}
2017-05-01 03:34:43 +02:00
} else if (path.startsWith(dir.path)) {
2017-02-20 18:37:11 +01:00
detectedItems << SyncthingItem(&dir, path.mid(dir.path.size()));
2017-02-25 18:28:20 +01:00
lastDir = &dir;
2017-05-01 03:34:43 +02:00
if (!containingDirs.contains(lastDir)) {
2017-02-25 18:28:20 +01:00
containingDirs << lastDir;
}
2017-02-20 18:37:11 +01:00
}
}
}
2017-02-25 18:28:20 +01:00
// add actions for the selected items itself
2017-05-01 03:34:43 +02:00
if (!detectedItems.isEmpty()) {
actions << new QAction(QIcon::fromTheme(QStringLiteral("view-refresh")),
detectedItems.size() == 1 ? tr("Rescan %1 (in %2)").arg(detectedItems.front().name, detectedItems.front().dir->displayName())
: tr("Rescan selected items"),
parentWidget);
if (s_connection.isConnected()) {
for (const SyncthingItem &item : detectedItems) {
2017-02-25 18:28:20 +01:00
connect(actions.back(), &QAction::triggered, bind(&SyncthingFileItemAction::rescanDir, item.dir->id, item.path));
}
} else {
actions.back()->setEnabled(false);
}
}
// add actions for explicitely selected Syncthing dirs
2017-05-01 03:34:43 +02:00
if (!detectedDirs.isEmpty()) {
2017-02-25 18:28:20 +01:00
// rescan item
2017-05-01 03:34:43 +02:00
actions << new QAction(QIcon::fromTheme(QStringLiteral("folder-sync")),
detectedDirs.size() == 1 ? tr("Rescan %1").arg(detectedDirs.front()->displayName()) : tr("Rescan selected directories"), parentWidget);
if (s_connection.isConnected()) {
for (const SyncthingDir *dir : detectedDirs) {
2017-02-20 18:37:11 +01:00
connect(actions.back(), &QAction::triggered, bind(&SyncthingFileItemAction::rescanDir, dir->id, QString()));
containingDirs.removeAll(dir);
}
} else {
actions.back()->setEnabled(false);
}
2017-02-25 18:28:20 +01:00
// pause/resume item
QStringList ids;
ids.reserve(detectedDirs.size());
bool isPaused = false;
2017-05-01 03:34:43 +02:00
for (const SyncthingDir *dir : detectedDirs) {
2017-02-25 18:28:20 +01:00
ids << dir->id;
2017-05-01 03:34:43 +02:00
if (dir->paused) {
2017-02-25 18:28:20 +01:00
isPaused = true;
break;
}
}
2017-05-01 03:34:43 +02:00
if (isPaused) {
actions << new QAction(QIcon::fromTheme(QStringLiteral("media-playback-start")),
detectedDirs.size() == 1 ? tr("Resume %1").arg(detectedDirs.front()->displayName()) : tr("Resume selected directories"),
parentWidget);
2017-02-25 18:28:20 +01:00
} else {
2017-05-01 03:34:43 +02:00
actions << new QAction(QIcon::fromTheme(QStringLiteral("media-playback-pause")),
detectedDirs.size() == 1 ? tr("Pause %1").arg(detectedDirs.front()->displayName()) : tr("Pause selected directories"), parentWidget);
2017-02-25 18:28:20 +01:00
}
2017-05-01 03:34:43 +02:00
if (s_connection.isConnected()) {
connect(actions.back(), &QAction::triggered,
bind(isPaused ? &SyncthingConnection::resumeDirectories : &SyncthingConnection::pauseDirectories, &s_connection, ids));
2017-02-25 18:28:20 +01:00
} else {
actions.back()->setEnabled(false);
}
2017-02-20 18:37:11 +01:00
}
// add actions for the Syncthing dirs containing selected items
2017-05-01 03:34:43 +02:00
if (!containingDirs.isEmpty()) {
2017-02-25 18:28:20 +01:00
// rescan item
2017-05-01 03:34:43 +02:00
actions << new QAction(QIcon::fromTheme(QStringLiteral("folder-sync")),
containingDirs.size() == 1 ? tr("Rescan %1").arg(containingDirs.front()->displayName()) : tr("Rescan containing directories"),
parentWidget);
if (s_connection.isConnected()) {
for (const SyncthingDir *dir : containingDirs) {
2017-02-20 18:37:11 +01:00
connect(actions.back(), &QAction::triggered, bind(&SyncthingFileItemAction::rescanDir, dir->id, QString()));
}
} else {
actions.back()->setEnabled(false);
}
2017-02-25 18:28:20 +01:00
// pause/resume item
QStringList ids;
ids.reserve(containingDirs.size());
bool isPaused = false;
2017-05-01 03:34:43 +02:00
for (const SyncthingDir *dir : containingDirs) {
2017-02-25 18:28:20 +01:00
ids << dir->id;
2017-05-01 03:34:43 +02:00
if (dir->paused) {
2017-02-25 18:28:20 +01:00
isPaused = true;
break;
2017-02-20 18:37:11 +01:00
}
2017-02-25 18:28:20 +01:00
}
2017-05-01 03:34:43 +02:00
if (isPaused) {
actions << new QAction(QIcon::fromTheme(QStringLiteral("media-playback-start")),
containingDirs.size() == 1 ? tr("Resume %1").arg(containingDirs.front()->displayName()) : tr("Resume containing directories"),
parentWidget);
2017-02-25 18:28:20 +01:00
} else {
2017-05-01 03:34:43 +02:00
actions << new QAction(QIcon::fromTheme(QStringLiteral("media-playback-pause")),
containingDirs.size() == 1 ? tr("Pause %1").arg(containingDirs.front()->displayName()) : tr("Pause containing directories"),
parentWidget);
2017-02-25 18:28:20 +01:00
}
2017-05-01 03:34:43 +02:00
if (s_connection.isConnected()) {
connect(actions.back(), &QAction::triggered,
bind(isPaused ? &SyncthingConnection::resumeDirectories : &SyncthingConnection::pauseDirectories, &s_connection, ids));
2017-02-20 18:37:11 +01:00
} else {
actions.back()->setEnabled(false);
}
}
// don't add any further actions if no relevant actions could be determined so far
2017-05-01 03:34:43 +02:00
if (actions.isEmpty()) {
2017-02-20 18:37:11 +01:00
return actions;
}
2017-03-13 00:40:48 +01:00
// add actions to show further information about directory if the selection is only about one particular Syncthing dir
2017-05-01 03:34:43 +02:00
if (detectedDirs.size() + containingDirs.size() == 1) {
2017-03-13 00:40:48 +01:00
auto *statusActions = new SyncthingDirActions(*lastDir, parentWidget);
2017-05-01 03:34:43 +02:00
connect(&s_connection, &SyncthingConnection::newDirs, statusActions,
static_cast<void (SyncthingDirActions::*)(const vector<SyncthingDir> &)>(&SyncthingDirActions::updateStatus));
connect(&s_connection, &SyncthingConnection::dirStatusChanged, statusActions,
static_cast<bool (SyncthingDirActions::*)(const SyncthingDir &)>(&SyncthingDirActions::updateStatus));
2017-03-13 00:40:48 +01:00
actions << *statusActions;
2017-02-20 18:37:11 +01:00
}
// about about action
QAction *separator = new QAction(parentWidget);
separator->setSeparator(true);
QAction *aboutAction = new QAction(QIcon::fromTheme(QStringLiteral("help-about")), tr("About"));
connect(aboutAction, &QAction::triggered, &SyncthingFileItemAction::showAboutDialog);
actions << separator << aboutAction;
2017-02-20 18:37:11 +01:00
return actions;
2017-02-20 18:37:11 +01:00
}
#include <syncthingfileitemaction.moc>