mirror of
https://github.com/transmission/transmission
synced 2024-12-23 16:24:02 +00:00
2b917de65b
This refactoring is driven by the need to be able to do true queued RPC calls (where each successive call uses the result of the previous). Currently, such queueing of requests is done by assigning them special "magic" tag numbers, which are then intercepted in one big switch() statement and acted upon. This (aside from making code greatly unclear) effectively makes each such queue a singleton, because state passing is restricted to global variables. We refactor RpcClient to assign an unique tag to each remote call, and then abstract all the call<->response matching with Qt's future/promise mechanism. Finally, we introduce a "RPC request queue" class (RpcQueue) which is built on top of QFutureWatcher and C++11's <functional> library. This class maintains a queue of functions, where each function receives an RPC response, does necessary processing, performs another call and finally returns its future.
100 lines
2 KiB
C++
100 lines
2 KiB
C++
/*
|
|
* This file Copyright (C) 2013-2016 Mnemosyne LLC
|
|
*
|
|
* It may be used under the GNU GPL versions 2 or 3
|
|
* or any future license endorsed by Mnemosyne LLC.
|
|
*
|
|
* $Id$
|
|
*/
|
|
|
|
#include <QDir>
|
|
|
|
#include <libtransmission/transmission.h>
|
|
#include <libtransmission/variant.h>
|
|
|
|
#include "Formatter.h"
|
|
#include "FreeSpaceLabel.h"
|
|
#include "RpcQueue.h"
|
|
#include "Session.h"
|
|
|
|
namespace
|
|
{
|
|
static const int INTERVAL_MSEC = 15000;
|
|
}
|
|
|
|
FreeSpaceLabel::FreeSpaceLabel (QWidget * parent):
|
|
QLabel (parent),
|
|
mySession (nullptr),
|
|
myTimer (this)
|
|
{
|
|
myTimer.setSingleShot (true);
|
|
myTimer.setInterval (INTERVAL_MSEC);
|
|
|
|
connect (&myTimer, SIGNAL (timeout ()), this, SLOT (onTimer ()));
|
|
}
|
|
|
|
void
|
|
FreeSpaceLabel::setSession (Session& session)
|
|
{
|
|
if (mySession == &session)
|
|
return;
|
|
|
|
mySession = &session;
|
|
onTimer ();
|
|
}
|
|
|
|
void
|
|
FreeSpaceLabel::setPath (const QString& path)
|
|
{
|
|
if (myPath != path)
|
|
{
|
|
setText (tr("<i>Calculating Free Space...</i>"));
|
|
myPath = path;
|
|
onTimer ();
|
|
}
|
|
}
|
|
|
|
void
|
|
FreeSpaceLabel::onTimer ()
|
|
{
|
|
myTimer.stop ();
|
|
|
|
if (mySession == nullptr || myPath.isEmpty ())
|
|
return;
|
|
|
|
tr_variant args;
|
|
tr_variantInitDict (&args, 1);
|
|
tr_variantDictAddStr (&args, TR_KEY_path, myPath.toUtf8 ().constData());
|
|
|
|
RpcQueue * q = new RpcQueue ();
|
|
|
|
q->add (
|
|
[this, &args] ()
|
|
{
|
|
return mySession->exec ("free-space", &args);
|
|
});
|
|
|
|
q->add (
|
|
[this] (const RpcResponse& r)
|
|
{
|
|
QString str;
|
|
|
|
// update the label
|
|
int64_t bytes = -1;
|
|
if (tr_variantDictFindInt (r.args.get (), TR_KEY_size_bytes, &bytes) && bytes >= 0)
|
|
setText (tr ("%1 free").arg (Formatter::sizeToString (bytes)));
|
|
else
|
|
setText (QString ());
|
|
|
|
// update the tooltip
|
|
size_t len = 0;
|
|
const char * path = 0;
|
|
tr_variantDictFindStr (r.args.get (), TR_KEY_path, &path, &len);
|
|
str = QString::fromUtf8 (path, len);
|
|
setToolTip (QDir::toNativeSeparators (str));
|
|
|
|
myTimer.start ();
|
|
});
|
|
|
|
q->run ();
|
|
}
|