1 /****************************************************************************
2 **
3 ** Copyright (C) 2015 The Qt Company Ltd.
4 ** Contact: http://www.qt.io/licensing/
5 **
6 ** This file is part of the QtLocation module of the Qt Toolkit.
7 **
8 ** $QT_BEGIN_LICENSE:LGPL3$
9 ** Commercial License Usage
10 ** Licensees holding valid commercial Qt licenses may use this file in
11 ** accordance with the commercial license agreement provided with the
12 ** Software or, alternatively, in accordance with the terms contained in
13 ** a written agreement between you and The Qt Company. For licensing terms
14 ** and conditions see http://www.qt.io/terms-conditions. For further
15 ** information use the contact form at http://www.qt.io/contact-us.
16 **
17 ** GNU Lesser General Public License Usage
18 ** Alternatively, this file may be used under the terms of the GNU Lesser
19 ** General Public License version 3 as published by the Free Software
20 ** Foundation and appearing in the file LICENSE.LGPLv3 included in the
21 ** packaging of this file. Please review the following information to
22 ** ensure the GNU Lesser General Public License version 3 requirements
23 ** will be met: https://www.gnu.org/licenses/lgpl.html.
24 **
25 ** GNU General Public License Usage
26 ** Alternatively, this file may be used under the terms of the GNU
27 ** General Public License version 2.0 or later as published by the Free
28 ** Software Foundation and appearing in the file LICENSE.GPL included in
29 ** the packaging of this file. Please review the following information to
30 ** ensure the GNU General Public License version 2.0 requirements will be
31 ** met: http://www.gnu.org/licenses/gpl-2.0.html.
32 **
33 ** $QT_END_LICENSE$
34 **
35 ****************************************************************************/
36 
37 #include <QtCore/QTimerEvent>
38 
39 #include "qgeomappingmanagerengine_p.h"
40 #include "qgeotilefetcher_p.h"
41 #include "qgeotilefetcher_p_p.h"
42 #include "qgeotiledmapreply_p.h"
43 #include "qgeotilespec_p.h"
44 #include "qgeotiledmap_p.h"
45 
46 #include <algorithm>
47 #include <iterator>
48 
49 QT_BEGIN_NAMESPACE
50 
QGeoTileFetcher(QGeoMappingManagerEngine * parent)51 QGeoTileFetcher::QGeoTileFetcher(QGeoMappingManagerEngine *parent)
52 :   QObject(*new QGeoTileFetcherPrivate(), parent)
53 {
54     Q_D(QGeoTileFetcher);
55 
56     d->enabled_ = true;
57     d->engine_ = parent;
58 }
59 
QGeoTileFetcher(QGeoTileFetcherPrivate & dd,QGeoMappingManagerEngine * parent)60 QGeoTileFetcher::QGeoTileFetcher(QGeoTileFetcherPrivate &dd, QGeoMappingManagerEngine *parent)
61 :   QObject(dd,parent)
62 {
63     Q_D(QGeoTileFetcher);
64     d->enabled_ = true;
65     d->engine_ = parent;
66 }
67 
~QGeoTileFetcher()68 QGeoTileFetcher::~QGeoTileFetcher()
69 {
70 }
71 
updateTileRequests(const QSet<QGeoTileSpec> & tilesAdded,const QSet<QGeoTileSpec> & tilesRemoved)72 void QGeoTileFetcher::updateTileRequests(const QSet<QGeoTileSpec> &tilesAdded,
73                                                   const QSet<QGeoTileSpec> &tilesRemoved)
74 {
75     Q_D(QGeoTileFetcher);
76 
77     QMutexLocker ml(&d->queueMutex_);
78 
79     cancelTileRequests(tilesRemoved);
80 
81     std::copy(tilesAdded.cbegin(), tilesAdded.cend(), std::back_inserter(d->queue_));
82 
83     if (d->enabled_ && initialized() && !d->queue_.isEmpty() && !d->timer_.isActive())
84         d->timer_.start(0, this);
85 }
86 
cancelTileRequests(const QSet<QGeoTileSpec> & tiles)87 void QGeoTileFetcher::cancelTileRequests(const QSet<QGeoTileSpec> &tiles)
88 {
89     Q_D(QGeoTileFetcher);
90 
91     typedef QSet<QGeoTileSpec>::const_iterator tile_iter;
92     // No need to lock: called only in updateTileRequests
93     tile_iter tile = tiles.constBegin();
94     tile_iter end = tiles.constEnd();
95     for (; tile != end; ++tile) {
96         QGeoTiledMapReply *reply = d->invmap_.value(*tile, 0);
97         if (reply) {
98             d->invmap_.remove(*tile);
99             reply->abort();
100             if (reply->isFinished())
101                 reply->deleteLater();
102         }
103         d->queue_.removeAll(*tile);
104     }
105 }
106 
requestNextTile()107 void QGeoTileFetcher::requestNextTile()
108 {
109     Q_D(QGeoTileFetcher);
110 
111     QMutexLocker ml(&d->queueMutex_);
112 
113     if (!d->enabled_)
114         return;
115 
116     if (d->queue_.isEmpty())
117         return;
118 
119     QGeoTileSpec ts = d->queue_.takeFirst();
120     if (d->queue_.isEmpty())
121         d->timer_.stop();
122 
123     // Check against min/max zoom to prevent sending requests for not existing objects
124     const QGeoCameraCapabilities & cameraCaps = d->engine_->cameraCapabilities(ts.mapId());
125     // the ZL in QGeoTileSpec is relative to the native tile size of the provider.
126     // It gets denormalized in QGeoTiledMap.
127     if (ts.zoom() < cameraCaps.minimumZoomLevel() || ts.zoom() > cameraCaps.maximumZoomLevel() || !fetchingEnabled())
128         return;
129 
130     QGeoTiledMapReply *reply = getTileImage(ts);
131     if (!reply)
132         return;
133 
134     if (reply->isFinished()) {
135         handleReply(reply, ts);
136     } else {
137         connect(reply,
138                 SIGNAL(finished()),
139                 this,
140                 SLOT(finished()),
141                 Qt::QueuedConnection);
142 
143         d->invmap_.insert(ts, reply);
144     }
145 }
146 
finished()147 void QGeoTileFetcher::finished()
148 {
149     Q_D(QGeoTileFetcher);
150 
151     QMutexLocker ml(&d->queueMutex_);
152 
153     QGeoTiledMapReply *reply = qobject_cast<QGeoTiledMapReply *>(sender());
154     if (!reply)
155         return;
156 
157     QGeoTileSpec spec = reply->tileSpec();
158 
159     if (!d->invmap_.contains(spec)) {
160         reply->deleteLater();
161         return;
162     }
163 
164     d->invmap_.remove(spec);
165 
166     handleReply(reply, spec);
167 }
168 
timerEvent(QTimerEvent * event)169 void QGeoTileFetcher::timerEvent(QTimerEvent *event)
170 {
171     Q_D(QGeoTileFetcher);
172     if (event->timerId() != d->timer_.timerId()) {
173         QObject::timerEvent(event);
174         return;
175     }
176 
177     QMutexLocker ml(&d->queueMutex_);
178     if (d->queue_.isEmpty() || !initialized()) {
179         d->timer_.stop();
180         return;
181     }
182     ml.unlock();
183 
184     requestNextTile();
185 }
186 
initialized() const187 bool QGeoTileFetcher::initialized() const
188 {
189     return true;
190 }
191 
fetchingEnabled() const192 bool QGeoTileFetcher::fetchingEnabled() const
193 {
194     return true;
195 }
196 
handleReply(QGeoTiledMapReply * reply,const QGeoTileSpec & spec)197 void QGeoTileFetcher::handleReply(QGeoTiledMapReply *reply, const QGeoTileSpec &spec)
198 {
199     Q_D(QGeoTileFetcher);
200 
201     if (!d->enabled_) {
202         reply->deleteLater();
203         return;
204     }
205 
206     if (reply->error() == QGeoTiledMapReply::NoError) {
207         emit tileFinished(spec, reply->mapImageData(), reply->mapImageFormat());
208     } else {
209         emit tileError(spec, reply->errorString());
210     }
211 
212     reply->deleteLater();
213 }
214 
215 /*******************************************************************************
216 *******************************************************************************/
217 
QGeoTileFetcherPrivate()218 QGeoTileFetcherPrivate::QGeoTileFetcherPrivate()
219 :   QObjectPrivate(), enabled_(false), engine_(0)
220 {
221 }
222 
~QGeoTileFetcherPrivate()223 QGeoTileFetcherPrivate::~QGeoTileFetcherPrivate()
224 {
225 }
226 
227 QT_END_NAMESPACE
228