blob: 344c0efe8bbffa7c579a99141471900080af640a (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
|
#pragma once
#include <QList>
#include <functional>
#include <QSharedPointer>
#include <QEventLoop>
#include <QDebug>
#include "resultprovider.h"
namespace async {
/*
* A result set specialization that provides a syncronous list.
*
* Only for testing purposes.
*
* WARNING: The nested eventloop can cause all sorts of trouble. Use only in testing code.
*/
template <class T>
class SyncListResult : public QList<T>
{
public:
SyncListResult(const QSharedPointer<Sink::ResultEmitter<T>> &emitter) : QList<T>(), mEmitter(emitter)
{
emitter->onAdded([this](const T &value) { this->append(value); });
emitter->onModified([this](const T &value) {
for (auto it = this->begin(); it != this->end(); it++) {
if (**it == *value) {
it = this->erase(it);
this->insert(it, value);
break;
}
}
});
emitter->onRemoved([this](const T &value) {
for (auto it = this->begin(); it != this->end(); it++) {
if (**it == *value) {
this->erase(it);
break;
}
}
});
emitter->onInitialResultSetComplete([this]() {
if (eventLoopAborter) {
eventLoopAborter();
// Be safe in case of a second invocation of the complete handler
eventLoopAborter = std::function<void()>();
}
});
emitter->onComplete([this]() { mEmitter.clear(); });
emitter->onClear([this]() { this->clear(); });
}
void exec()
{
QEventLoop eventLoop;
eventLoopAborter = [&eventLoop]() { eventLoop.quit(); };
eventLoop.exec();
}
private:
QSharedPointer<Sink::ResultEmitter<T>> mEmitter;
std::function<void()> eventLoopAborter;
};
}
|