1
0
mirror of https://github.com/tumic0/GPXSee.git synced 2024-11-30 22:51:16 +01:00
GPXSee/src/map/mbtilesmap.h
2023-12-25 11:12:52 +01:00

165 lines
3.6 KiB
C++

#ifndef MBTILESMAP_H
#define MBTILESMAP_H
#include <QDebug>
#include <QSqlDatabase>
#include <QVector>
#include <QImageReader>
#include <QBuffer>
#include <QPixmap>
#include <QtConcurrent>
#include "map.h"
class MBTile
{
public:
MBTile(int zoom, int overzoom, int scaledSize, const QPoint &xy,
const QByteArray &data, const QString &key) : _zoom(zoom),
_overzoom(overzoom), _scaledSize(scaledSize), _xy(xy), _data(data),
_key(key) {}
const QPoint &xy() const {return _xy;}
const QString &key() const {return _key;}
const QPixmap &pixmap() const {return _pixmap;}
void load() {
QByteArray format(_overzoom
? QByteArray::number(_zoom) + ';' + QByteArray::number(_overzoom)
: QByteArray::number(_zoom));
QBuffer buffer(&_data);
QImageReader reader(&buffer, format);
if (_scaledSize)
reader.setScaledSize(QSize(_scaledSize, _scaledSize));
_pixmap = QPixmap::fromImage(reader.read());
}
private:
int _zoom;
int _overzoom;
int _scaledSize;
QPoint _xy;
QByteArray _data;
QString _key;
QPixmap _pixmap;
};
class MBTilesMapJob : public QObject
{
Q_OBJECT
public:
MBTilesMapJob(const QList<MBTile> &tiles) : _tiles(tiles) {}
void run()
{
connect(&_watcher, &QFutureWatcher<void>::finished, this,
&MBTilesMapJob::handleFinished);
_future = QtConcurrent::map(_tiles, &MBTile::load);
_watcher.setFuture(_future);
}
void cancel(bool wait)
{
_future.cancel();
if (wait)
_future.waitForFinished();
}
const QList<MBTile> &tiles() const {return _tiles;}
signals:
void finished(MBTilesMapJob *job);
private slots:
void handleFinished() {emit finished(this);}
private:
QFutureWatcher<void> _watcher;
QFuture<void> _future;
QList<MBTile> _tiles;
};
class MBTilesMap : public Map
{
Q_OBJECT
public:
MBTilesMap(const QString &fileName, QObject *parent = 0);
QString name() const {return _name;}
QRectF bounds();
RectC llBounds() {return _bounds;}
qreal resolution(const QRectF &rect);
int zoom() const {return _zi;}
void setZoom(int zoom) {_zi = zoom;}
int zoomFit(const QSize &size, const RectC &rect);
int zoomIn();
int zoomOut();
QPointF ll2xy(const Coordinates &c);
Coordinates xy2ll(const QPointF &p);
void draw(QPainter *painter, const QRectF &rect, Flags flags);
void load(const Projection &in, const Projection &out, qreal deviceRatio,
bool hidpi);
void unload();
bool isValid() const {return _valid;}
QString errorString() const {return _errorString;}
static Map *create(const QString &path, const Projection &proj, bool *isDir);
private slots:
void jobFinished(MBTilesMapJob *job);
private:
struct Zoom {
Zoom() : z(-1), base(-1) {}
Zoom(int z, int base) : z(z), base(base) {}
int z;
int base;
};
bool getMinZoom(int &zoom);
bool getMaxZoom(int &zoom);
bool getZooms();
bool getBounds();
bool getTileSize();
void getTileFormat();
void getTilePixelRatio();
void getName();
qreal tileSize() const;
qreal coordinatesRatio() const;
qreal imageRatio() const;
QByteArray tileData(int zoom, const QPoint &tile) const;
void drawTile(QPainter *painter, QPixmap &pixmap, QPointF &tp);
bool isRunning(const QString &key) const;
void runJob(MBTilesMapJob *job);
void removeJob(MBTilesMapJob *job);
void cancelJobs(bool wait);
QPointF tilePos(const QPointF &tl, const QPoint &tc, const QPoint &tile,
unsigned overzoom) const;
friend QDebug operator<<(QDebug dbg, const Zoom &zoom);
QSqlDatabase _db;
QString _name;
RectC _bounds;
QVector<Zoom> _zooms, _zoomsBase;
int _zi;
int _tileSize;
qreal _mapRatio, _tileRatio;
bool _scalable;
int _scaledSize;
QList<MBTilesMapJob*> _jobs;
bool _valid;
QString _errorString;
};
#endif // MBTILESMAP_H