mirror of
https://github.com/tumic0/GPXSee.git
synced 2024-12-03 16:09:08 +01:00
Compare commits
3 Commits
148fc76d5a
...
c1c8f6303c
Author | SHA1 | Date | |
---|---|---|---|
c1c8f6303c | |||
5fcc27c176 | |||
77e9fae19d |
@ -1,7 +1,6 @@
|
||||
#include <QPainter>
|
||||
#include <QDir>
|
||||
#include <QPixmapCache>
|
||||
#include <QtConcurrent>
|
||||
#include "common/rectc.h"
|
||||
#include "common/programpaths.h"
|
||||
#include "common/downloader.h"
|
||||
@ -11,18 +10,13 @@
|
||||
|
||||
#define MAX_OVERZOOM 3
|
||||
|
||||
static QString cacheName(const QString &file, unsigned overzoom)
|
||||
{
|
||||
return overzoom ? file + ":" + QString::number(overzoom) : file;
|
||||
}
|
||||
|
||||
OnlineMap::OnlineMap(const QString &fileName, const QString &name,
|
||||
const QString &url, const Range &zooms, const RectC &bounds, qreal tileRatio,
|
||||
const QList<HTTPHeader> &headers, int tileSize, bool scalable, bool invertY,
|
||||
bool quadTiles, QObject *parent)
|
||||
: Map(fileName, parent), _name(name), _zooms(zooms), _bounds(bounds),
|
||||
_zoom(_zooms.max()), _tileSize(tileSize), _base(0), _mapRatio(1.0),
|
||||
_tileRatio(tileRatio), _scalable(scalable), _invertY(invertY)
|
||||
_zoom(_zooms.max()), _tileSize(tileSize), _baseZoom(0), _mapRatio(1.0),
|
||||
_tileRatio(tileRatio), _scalable(scalable), _scaledSize(0), _invertY(invertY)
|
||||
{
|
||||
_tileLoader = new TileLoader(QDir(ProgramPaths::tilesDir()).filePath(_name),
|
||||
this);
|
||||
@ -31,7 +25,7 @@ OnlineMap::OnlineMap(const QString &fileName, const QString &name,
|
||||
connect(_tileLoader, &TileLoader::finished, this, &OnlineMap::tilesLoaded);
|
||||
|
||||
if (_scalable) {
|
||||
_base = _zooms.max();
|
||||
_baseZoom = _zooms.max();
|
||||
_zooms.setMax(qMin(_zooms.max() + MAX_OVERZOOM, OSM::ZOOMS.max()));
|
||||
}
|
||||
}
|
||||
@ -72,12 +66,16 @@ qreal OnlineMap::resolution(const QRectF &rect)
|
||||
|
||||
int OnlineMap::zoomIn()
|
||||
{
|
||||
cancelJobs(false);
|
||||
|
||||
_zoom = qMin(_zoom + 1, _zooms.max());
|
||||
return _zoom;
|
||||
}
|
||||
|
||||
int OnlineMap::zoomOut()
|
||||
{
|
||||
cancelJobs(false);
|
||||
|
||||
_zoom = qMax(_zoom - 1, _zooms.min());
|
||||
return _zoom;
|
||||
}
|
||||
@ -96,6 +94,11 @@ void OnlineMap::load(const Projection &in, const Projection &out,
|
||||
}
|
||||
}
|
||||
|
||||
void OnlineMap::unload()
|
||||
{
|
||||
cancelJobs(true);
|
||||
}
|
||||
|
||||
qreal OnlineMap::coordinatesRatio() const
|
||||
{
|
||||
return _mapRatio > 1.0 ? _mapRatio / _tileRatio : 1.0;
|
||||
@ -116,16 +119,63 @@ QPoint OnlineMap::tileCoordinates(int x, int y, int zoom)
|
||||
return QPoint(x, _invertY ? (1<<zoom) - y - 1 : y);
|
||||
}
|
||||
|
||||
bool OnlineMap::isRunning(const QString &key) const
|
||||
{
|
||||
for (int i = 0; i < _jobs.size(); i++) {
|
||||
const QList<OnlineMapTile> &tiles = _jobs.at(i)->tiles();
|
||||
for (int j = 0; j < tiles.size(); j++)
|
||||
if (tiles.at(j).key() == key)
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
void OnlineMap::runJob(OnlineMapJob *job)
|
||||
{
|
||||
_jobs.append(job);
|
||||
|
||||
connect(job, &OnlineMapJob::finished, this, &OnlineMap::jobFinished);
|
||||
job->run();
|
||||
}
|
||||
|
||||
void OnlineMap::removeJob(OnlineMapJob *job)
|
||||
{
|
||||
_jobs.removeOne(job);
|
||||
job->deleteLater();
|
||||
}
|
||||
|
||||
void OnlineMap::jobFinished(OnlineMapJob *job)
|
||||
{
|
||||
const QList<OnlineMapTile> &tiles = job->tiles();
|
||||
|
||||
for (int i = 0; i < tiles.size(); i++) {
|
||||
const OnlineMapTile &mt = tiles.at(i);
|
||||
if (!mt.pixmap().isNull())
|
||||
QPixmapCache::insert(mt.key(), mt.pixmap());
|
||||
}
|
||||
|
||||
removeJob(job);
|
||||
|
||||
emit tilesLoaded();
|
||||
}
|
||||
|
||||
void OnlineMap::cancelJobs(bool wait)
|
||||
{
|
||||
for (int i = 0; i < _jobs.size(); i++)
|
||||
_jobs.at(i)->cancel(wait);
|
||||
}
|
||||
|
||||
void OnlineMap::draw(QPainter *painter, const QRectF &rect, Flags flags)
|
||||
{
|
||||
int base = _scalable ? qMin(_base, _zoom) : _zoom;
|
||||
unsigned overzoom = _zoom - base;
|
||||
int baseZoom = _scalable ? qMin(_baseZoom, _zoom) : _zoom;
|
||||
unsigned overzoom = _zoom - baseZoom;
|
||||
unsigned f = 1U<<overzoom;
|
||||
|
||||
qreal scale = OSM::zoom2scale(base, _tileSize * f);
|
||||
qreal scale = OSM::zoom2scale(baseZoom, _tileSize * f);
|
||||
QPoint tile = OSM::mercator2tile(QPointF(rect.topLeft().x() * scale,
|
||||
-rect.topLeft().y() * scale) * coordinatesRatio(), base);
|
||||
Coordinates ctl(OSM::tile2ll(tile, base));
|
||||
-rect.topLeft().y() * scale) * coordinatesRatio(), baseZoom);
|
||||
Coordinates ctl(OSM::tile2ll(tile, baseZoom));
|
||||
QPointF tl(ll2xy(Coordinates(ctl.lon(), -ctl.lat())));
|
||||
QSizeF s(rect.right() - tl.x(), rect.bottom() - tl.y());
|
||||
int width = ceil(s.width() / (tileSize() * f));
|
||||
@ -135,8 +185,8 @@ void OnlineMap::draw(QPainter *painter, const QRectF &rect, Flags flags)
|
||||
fetchTiles.reserve(width * height);
|
||||
for (int i = 0; i < width; i++) {
|
||||
for (int j = 0; j < height; j++) {
|
||||
QPoint tc(tileCoordinates(tile.x() + i, tile.y() + j, base));
|
||||
fetchTiles.append(TileLoader::Tile(tc, base));
|
||||
QPoint tc(tileCoordinates(tile.x() + i, tile.y() + j, baseZoom));
|
||||
fetchTiles.append(TileLoader::Tile(tc, baseZoom));
|
||||
}
|
||||
}
|
||||
|
||||
@ -145,37 +195,50 @@ void OnlineMap::draw(QPainter *painter, const QRectF &rect, Flags flags)
|
||||
else
|
||||
_tileLoader->loadTilesAsync(fetchTiles);
|
||||
|
||||
QList<OnlineTile> renderTiles;
|
||||
QList<OnlineMapTile> renderTiles;
|
||||
for (int i = 0; i < fetchTiles.count(); i++) {
|
||||
const TileLoader::Tile &t = fetchTiles.at(i);
|
||||
if (t.file().isNull())
|
||||
continue;
|
||||
|
||||
QString key(overzoom
|
||||
? t.file() + ":" + QString::number(overzoom) : t.file());
|
||||
if (isRunning(key))
|
||||
continue;
|
||||
|
||||
QPixmap pm;
|
||||
if (QPixmapCache::find(cacheName(t.file(), overzoom), &pm)) {
|
||||
QPointF tp(tl.x() + (t.xy().x() - tile.x()) * tileSize() * f,
|
||||
tl.y() + (t.xy().y() - tile.y()) * tileSize() * f);
|
||||
if (QPixmapCache::find(key, &pm)) {
|
||||
QPoint tc(tileCoordinates(t.xy().x(), t.xy().y(), baseZoom));
|
||||
QPointF tp(tl.x() + (tc.x() - tile.x()) * tileSize() * f,
|
||||
tl.y() + (tc.y() - tile.y()) * tileSize() * f);
|
||||
drawTile(painter, pm, tp);
|
||||
} else
|
||||
renderTiles.append(OnlineTile(t.xy(), t.file(), _zoom, overzoom,
|
||||
_scaledSize));
|
||||
renderTiles.append(OnlineMapTile(t.xy(), t.file(), _zoom, overzoom,
|
||||
_scaledSize, key));
|
||||
}
|
||||
|
||||
QFuture<void> future = QtConcurrent::map(renderTiles, &OnlineTile::load);
|
||||
if (!renderTiles.isEmpty()) {
|
||||
if (flags & Map::Block || !_scalable) {
|
||||
QFuture<void> future = QtConcurrent::map(renderTiles,
|
||||
&OnlineMapTile::load);
|
||||
future.waitForFinished();
|
||||
|
||||
for (int i = 0; i < renderTiles.size(); i++) {
|
||||
const OnlineTile &mt = renderTiles.at(i);
|
||||
const OnlineMapTile &mt = renderTiles.at(i);
|
||||
QPixmap pm(mt.pixmap());
|
||||
if (pm.isNull())
|
||||
continue;
|
||||
|
||||
QPixmapCache::insert(cacheName(mt.file(), overzoom), pm);
|
||||
QPixmapCache::insert(mt.key(), pm);
|
||||
|
||||
QPointF tp(tl.x() + (mt.xy().x() - tile.x()) * tileSize() * f,
|
||||
tl.y() + (mt.xy().y() - tile.y()) * tileSize() * f);
|
||||
QPoint tc(tileCoordinates(mt.xy().x(), mt.xy().y(), baseZoom));
|
||||
QPointF tp(tl.x() + (tc.x() - tile.x()) * tileSize() * f,
|
||||
tl.y() + (tc.y() - tile.y()) * tileSize() * f);
|
||||
drawTile(painter, pm, tp);
|
||||
}
|
||||
} else
|
||||
runJob(new OnlineMapJob(renderTiles));
|
||||
}
|
||||
}
|
||||
|
||||
void OnlineMap::drawTile(QPainter *painter, QPixmap &pixmap, QPointF &tp)
|
||||
|
@ -3,17 +3,18 @@
|
||||
|
||||
#include <QImageReader>
|
||||
#include <QPixmap>
|
||||
#include <QtConcurrent>
|
||||
#include "common/range.h"
|
||||
#include "common/rectc.h"
|
||||
#include "map.h"
|
||||
#include "tileloader.h"
|
||||
|
||||
class OnlineTile
|
||||
class OnlineMapTile
|
||||
{
|
||||
public:
|
||||
OnlineTile(const QPoint &xy, const QString &file, int zoom, int overzoom,
|
||||
int scaledSize) : _xy(xy), _file(file), _zoom(zoom), _overzoom(overzoom),
|
||||
_scaledSize(scaledSize) {}
|
||||
OnlineMapTile(const QPoint &xy, const QString &file, int zoom, int overzoom,
|
||||
int scaledSize, const QString &key) : _zoom(zoom), _overzoom(overzoom),
|
||||
_scaledSize(scaledSize), _xy(xy), _file(file), _key(key) {}
|
||||
|
||||
void load()
|
||||
{
|
||||
@ -27,18 +28,53 @@ public:
|
||||
}
|
||||
|
||||
const QPoint &xy() const {return _xy;}
|
||||
const QString &file() const {return _file;}
|
||||
const QPixmap &pixmap() const {return _pixmap;}
|
||||
const QString &key() const {return _key;}
|
||||
|
||||
private:
|
||||
QPoint _xy;
|
||||
QString _file;
|
||||
int _zoom;
|
||||
int _overzoom;
|
||||
int _scaledSize;
|
||||
QPoint _xy;
|
||||
QString _file;
|
||||
QString _key;
|
||||
QPixmap _pixmap;
|
||||
};
|
||||
|
||||
class OnlineMapJob : public QObject
|
||||
{
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
OnlineMapJob(const QList<OnlineMapTile> &tiles) : _tiles(tiles) {}
|
||||
|
||||
void run()
|
||||
{
|
||||
connect(&_watcher, &QFutureWatcher<void>::finished, this,
|
||||
&OnlineMapJob::handleFinished);
|
||||
_future = QtConcurrent::map(_tiles, &OnlineMapTile::load);
|
||||
_watcher.setFuture(_future);
|
||||
}
|
||||
void cancel(bool wait)
|
||||
{
|
||||
_future.cancel();
|
||||
if (wait)
|
||||
_future.waitForFinished();
|
||||
}
|
||||
const QList<OnlineMapTile> &tiles() const {return _tiles;}
|
||||
|
||||
signals:
|
||||
void finished(OnlineMapJob *job);
|
||||
|
||||
private slots:
|
||||
void handleFinished() {emit finished(this);}
|
||||
|
||||
private:
|
||||
QFutureWatcher<void> _watcher;
|
||||
QFuture<void> _future;
|
||||
QList<OnlineMapTile> _tiles;
|
||||
};
|
||||
|
||||
class OnlineMap : public Map
|
||||
{
|
||||
Q_OBJECT
|
||||
@ -68,8 +104,12 @@ public:
|
||||
|
||||
void load(const Projection &in, const Projection &out, qreal deviceRatio,
|
||||
bool hidpi);
|
||||
void unload();
|
||||
void clearCache();
|
||||
|
||||
private slots:
|
||||
void jobFinished(OnlineMapJob *job);
|
||||
|
||||
private:
|
||||
int limitZoom(int zoom) const;
|
||||
qreal tileSize() const;
|
||||
@ -77,6 +117,10 @@ private:
|
||||
qreal imageRatio() const;
|
||||
QPoint tileCoordinates(int x, int y, int zoom);
|
||||
void drawTile(QPainter *painter, QPixmap &pixmap, QPointF &tp);
|
||||
bool isRunning(const QString &key) const;
|
||||
void runJob(OnlineMapJob *job);
|
||||
void removeJob(OnlineMapJob *job);
|
||||
void cancelJobs(bool wait);
|
||||
|
||||
TileLoader *_tileLoader;
|
||||
QString _name;
|
||||
@ -84,11 +128,13 @@ private:
|
||||
RectC _bounds;
|
||||
int _zoom;
|
||||
int _tileSize;
|
||||
int _base;
|
||||
int _baseZoom;
|
||||
qreal _mapRatio, _tileRatio;
|
||||
bool _scalable;
|
||||
int _scaledSize;
|
||||
bool _invertY;
|
||||
|
||||
QList<OnlineMapJob*> _jobs;
|
||||
};
|
||||
|
||||
#endif // ONLINEMAP_H
|
||||
|
Loading…
Reference in New Issue
Block a user