Skip to content
This repository has been archived by the owner on Sep 4, 2021. It is now read-only.

Latest commit

 

History

History
271 lines (241 loc) · 9.57 KB

synchronizelists.h

File metadata and controls

271 lines (241 loc) · 9.57 KB
 
Jul 16, 2013
Jul 16, 2013
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
/*
* Copyright (C) 2013 Jolla Mobile <andrew.den.exter@jollamobile.com>
*
* You may use this file under the terms of the BSD license as follows:
*
* "Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Nemo Mobile nor the names of its contributors
* may be used to endorse or promote products derived from this
* software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
*/
Jul 18, 2013
Jul 18, 2013
32
33
#ifndef SYNCHRONIZELISTS_H
#define SYNCHRONIZELISTS_H
Jul 16, 2013
Jul 16, 2013
34
35
36
37
// Helper utility to synchronize a cached list with some reference list with correct
// QAbstractItemModel signals and filtering.
Jul 18, 2013
Jul 18, 2013
38
39
40
41
42
// If the reference list is populated incrementally synchronizeList can be called multiple times with the
// same variables c and r to progressively synchronize the lists. After the final call completeSynchronizeList
// can be called to remove or append any items which remain unsynchronized.
// The Filtered variants allow the reference list to be filtered by a callback function to
// exclude unwanted items from the synchronized list.
Jul 16, 2013
Jul 16, 2013
43
Jul 18, 2013
Jul 18, 2013
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
template <typename T>
bool compareIdentity(const T &item, const T &reference)
{
return item == reference;
}
template <typename Agent, typename ReferenceList>
int insertRange(Agent *agent, int index, int count, const ReferenceList &source, int sourceIndex)
{
agent->insertRange(index, count, source, sourceIndex);
return count;
}
template <typename Agent>
int removeRange(Agent *agent, int index, int count)
{
agent->removeRange(index, count);
return 0;
}
template <typename Agent, typename ReferenceList>
int updateRange(Agent *agent, int index, int count, const ReferenceList &source, int sourceIndex)
{
Q_UNUSED(agent);
Q_UNUSED(index);
Q_UNUSED(source);
Q_UNUSED(sourceIndex);
return count;
}
template <typename Agent, typename CacheList, typename ReferenceList>
Jul 16, 2013
Jul 16, 2013
75
76
class SynchronizeList
{
Jul 18, 2013
Jul 18, 2013
77
78
typedef typename CacheList::const_reference CacheItem;
typedef typename ReferenceList::const_reference ReferenceItem;
Jul 16, 2013
Jul 16, 2013
79
80
81
82
public:
SynchronizeList(
Agent *agent,
Jul 18, 2013
Jul 18, 2013
83
const CacheList &cache,
Jul 16, 2013
Jul 16, 2013
84
85
86
87
88
int &c,
const ReferenceList &reference,
int &r)
: agent(agent), cache(cache), c(c), reference(reference), r(r)
{
Jul 18, 2013
Jul 18, 2013
89
90
91
92
int lastEqualC = c;
int lastEqualR = r;
for (; c < cache.count() && r < reference.count(); ++c, ++r) {
if (compareIdentity(cache.at(c), reference.at(r))) {
Jul 16, 2013
Jul 16, 2013
93
94
95
continue;
}
Jul 18, 2013
Jul 18, 2013
96
97
98
99
100
101
if (c > lastEqualC) {
lastEqualC += updateRange(agent, lastEqualC, c - lastEqualC, reference, lastEqualR);
c = lastEqualC;
lastEqualR = r;
}
Jul 16, 2013
Jul 16, 2013
102
103
104
105
106
107
108
bool match = false;
// Iterate through both the reference and cache lists in parallel looking for first
// point of commonality, when that is found resolve the differences and continue
// looking.
int count = 1;
for (; !match && c + count < cache.count() && r + count < reference.count(); ++count) {
Jul 18, 2013
Jul 18, 2013
109
110
CacheItem cacheItem = cache.at(c + count);
ReferenceItem referenceItem = reference.at(r + count);
Jul 16, 2013
Jul 16, 2013
111
112
for (int i = 0; i <= count; ++i) {
Jul 18, 2013
Jul 18, 2013
113
if (cacheMatch(i, count, referenceItem) || referenceMatch(i, count, cacheItem)) {
Jul 16, 2013
Jul 16, 2013
114
115
116
117
118
119
120
121
match = true;
break;
}
}
}
// Continue scanning the reference list if the cache has been exhausted.
for (int re = r + count; !match && re < reference.count(); ++re) {
Jul 18, 2013
Jul 18, 2013
122
ReferenceItem referenceItem = reference.at(re);
Jul 16, 2013
Jul 16, 2013
123
for (int i = 0; i < count; ++i) {
Jul 18, 2013
Jul 18, 2013
124
if (cacheMatch(i, re - r, referenceItem)) {
Jul 16, 2013
Jul 16, 2013
125
126
127
128
129
130
131
132
match = true;
break;
}
}
}
// Continue scanning the cache if the reference list has been exhausted.
for (int ce = c + count; !match && ce < cache.count(); ++ce) {
Jul 18, 2013
Jul 18, 2013
133
CacheItem cacheItem = cache.at(ce);
Jul 16, 2013
Jul 16, 2013
134
for (int i = 0; i < count; ++i) {
Jul 18, 2013
Jul 18, 2013
135
if (referenceMatch(i, ce - c, cacheItem)) {
Jul 16, 2013
Jul 16, 2013
136
137
138
139
140
141
142
143
match = true;
break;
}
}
}
if (!match)
return;
Jul 18, 2013
Jul 18, 2013
144
145
146
147
148
149
150
lastEqualC = c;
lastEqualR = r;
}
if (c > lastEqualC) {
updateRange(agent, lastEqualC, c - lastEqualC, reference, lastEqualR);
Jul 16, 2013
Jul 16, 2013
151
152
153
154
}
}
private:
Jul 18, 2013
Jul 18, 2013
155
// Tests if the cached contact id at i matches a referenceId.
Jul 16, 2013
Jul 16, 2013
156
157
// If there is a match removes all items traversed in the cache since the previous match
// and inserts any items in the reference set found to to not be in the cache.
Jul 18, 2013
Jul 18, 2013
158
bool cacheMatch(int i, int count, ReferenceItem referenceItem)
Jul 16, 2013
Jul 16, 2013
159
{
Jul 18, 2013
Jul 18, 2013
160
if (compareIdentity(cache.at(c + i), referenceItem)) {
Jul 16, 2013
Jul 16, 2013
161
if (i > 0)
Jul 18, 2013
Jul 18, 2013
162
163
164
c += removeRange(agent, c, i);
c += insertRange(agent, c, count, reference, r);
r += count;
Jul 16, 2013
Jul 16, 2013
165
166
167
168
169
170
return true;
} else {
return false;
}
}
Jul 18, 2013
Jul 18, 2013
171
// Tests if the reference contact id at i matches a cacheId.
Jul 16, 2013
Jul 16, 2013
172
173
174
// If there is a match inserts all items traversed in the reference set since the
// previous match and removes any items from the cache that were not found in the
// reference list.
Jul 18, 2013
Jul 18, 2013
175
bool referenceMatch(int i, int count, CacheItem cacheItem)
Jul 16, 2013
Jul 16, 2013
176
{
Jul 18, 2013
Jul 18, 2013
177
178
if (compareIdentity(reference.at(r + i), cacheItem)) {
c += removeRange(agent, c, count);
Jul 16, 2013
Jul 16, 2013
179
if (i > 0)
Jul 18, 2013
Jul 18, 2013
180
181
c += insertRange(agent, c, i, reference, r);
r += i;
Jul 16, 2013
Jul 16, 2013
182
183
184
185
186
187
188
return true;
} else {
return false;
}
}
Agent * const agent;
Jul 18, 2013
Jul 18, 2013
189
const CacheList &cache;
Jul 16, 2013
Jul 16, 2013
190
191
192
193
194
int &c;
const ReferenceList &reference;
int &r;
};
Jul 18, 2013
Jul 18, 2013
195
196
template <typename Agent, typename CacheList, typename ReferenceList>
void completeSynchronizeList(
Jul 16, 2013
Jul 16, 2013
197
Agent *agent,
Jul 18, 2013
Jul 18, 2013
198
199
const CacheList &cache,
int &cacheIndex,
Jul 16, 2013
Jul 16, 2013
200
const ReferenceList &reference,
Jul 18, 2013
Jul 18, 2013
201
int &referenceIndex)
Jul 16, 2013
Jul 16, 2013
202
{
Jul 18, 2013
Jul 18, 2013
203
204
if (cacheIndex < cache.count()) {
agent->removeRange(cacheIndex, cache.count() - cacheIndex);
Jul 16, 2013
Jul 16, 2013
205
}
Jul 18, 2013
Jul 18, 2013
206
207
if (referenceIndex < reference.count()) {
agent->insertRange(cache.count(), reference.count() - referenceIndex, reference, referenceIndex);
Jul 16, 2013
Jul 16, 2013
208
209
}
Jul 18, 2013
Jul 18, 2013
210
211
212
cacheIndex = 0;
referenceIndex = 0;
}
Jul 16, 2013
Jul 16, 2013
213
Jul 18, 2013
Jul 18, 2013
214
215
216
217
218
219
220
221
222
223
224
template <typename Agent, typename CacheList, typename ReferenceList>
void synchronizeList(
Agent *agent,
const CacheList &cache,
int &cacheIndex,
const ReferenceList &reference,
int &referenceIndex)
{
SynchronizeList<Agent, CacheList, ReferenceList>(
agent, cache, cacheIndex, reference, referenceIndex);
}
Jul 16, 2013
Jul 16, 2013
225
Jul 18, 2013
Jul 18, 2013
226
227
228
229
230
231
232
233
template <typename Agent, typename CacheList, typename ReferenceList>
void synchronizeList(Agent *agent, const CacheList &cache, const ReferenceList &reference)
{
int cacheIndex = 0;
int referenceIndex = 0;
synchronizeList(agent, cache, cacheIndex, reference, referenceIndex);
completeSynchronizeList(agent, cache, cacheIndex, reference, referenceIndex);
}
Jul 16, 2013
Jul 16, 2013
234
Jul 18, 2013
Jul 18, 2013
235
236
237
238
239
240
241
242
243
244
template <typename Agent, typename ReferenceList>
ReferenceList filterList(
Agent *agent,
const ReferenceList &reference)
{
ReferenceList filtered;
filtered.reserve(reference.count());
foreach (const typename ReferenceList::value_type &value, reference)
if (agent->filterValue(value))
filtered.append(value);
Jul 16, 2013
Jul 16, 2013
245
Jul 18, 2013
Jul 18, 2013
246
247
return filtered;
}
Jul 16, 2013
Jul 16, 2013
248
Jul 18, 2013
Jul 18, 2013
249
template <typename Agent, typename CacheList, typename ReferenceList>
Jul 16, 2013
Jul 16, 2013
250
251
void synchronizeFilteredList(
Agent *agent,
Jul 18, 2013
Jul 18, 2013
252
253
const CacheList &cache,
int &cacheIndex,
Jul 16, 2013
Jul 16, 2013
254
const ReferenceList &reference,
Jul 18, 2013
Jul 18, 2013
255
256
257
258
259
260
261
262
int &referenceIndex)
{
ReferenceList filtered = filterList(agent, reference);
synchronizeList(agent, cache, cacheIndex, filtered, referenceIndex);
}
template <typename Agent, typename CacheList, typename ReferenceList>
void synchronizeFilteredList(Agent *agent, const CacheList &cache, const ReferenceList &reference)
Jul 16, 2013
Jul 16, 2013
263
{
Jul 18, 2013
Jul 18, 2013
264
265
266
267
268
int cacheIndex = 0;
int referenceIndex = 0;
ReferenceList filtered = filterList(agent, reference);
synchronizeList(agent, cache, cacheIndex, filtered, referenceIndex);
completeSynchronizeList(agent, cache, cacheIndex, filtered, referenceIndex);
Jul 16, 2013
Jul 16, 2013
269
270
271
}
#endif