#include #include #include #include #include #include #if QT_VERSION < QT_VERSION_CHECK(5, 0, 0) #include #else // QT_VERSION < 5 #include #endif // QT_VERSION < 5 #include "common/rectc.h" #include "common/config.h" #include "osm.h" #include "mbtilesmap.h" class MBTile { public: MBTile(int zoom, const QPoint &xy, const QByteArray &data, const QString &key) : _zoom(zoom), _xy(xy), _data(data), _key(key) {} const QPoint &xy() const {return _xy;} const QString &key() const {return _key;} QPixmap pixmap() const {return QPixmap::fromImage(_image);} void load() { QByteArray z(QString::number(_zoom).toLatin1()); _image.loadFromData(_data, z); } private: int _zoom; QPoint _xy; QByteArray _data; QString _key; QImage _image; }; #define META_TYPE(type) static_cast(type) static void render(MBTile &tile) { tile.load(); } static double index2mercator(int index, int zoom) { return rad2deg(-M_PI + 2 * M_PI * ((double)index / (1< _zooms.max()) return _zooms.max(); return zoom; } int MBTilesMap::zoomFit(const QSize &size, const RectC &rect) { if (!rect.isValid()) _zoom = _zooms.max(); else { QRectF tbr(OSM::ll2m(rect.topLeft()), OSM::ll2m(rect.bottomRight())); QPointF sc(tbr.width() / size.width(), tbr.height() / size.height()); _zoom = limitZoom(OSM::scale2zoom(qMax(sc.x(), -sc.y()) / coordinatesRatio(), _tileSize)); } return _zoom; } qreal MBTilesMap::resolution(const QRectF &rect) { return OSM::resolution(rect.center(), _zoom, _tileSize); } int MBTilesMap::zoomIn() { _zoom = qMin(_zoom + 1, _zooms.max()); return _zoom; } int MBTilesMap::zoomOut() { _zoom = qMax(_zoom - 1, _zooms.min()); return _zoom; } qreal MBTilesMap::coordinatesRatio() const { return _deviceRatio > 1.0 ? _deviceRatio / _tileRatio : 1.0; } qreal MBTilesMap::imageRatio() const { return _deviceRatio > 1.0 ? _deviceRatio : _tileRatio; } qreal MBTilesMap::tileSize() const { return (_tileSize / coordinatesRatio()); } QByteArray MBTilesMap::tileData(int zoom, const QPoint &tile) const { QSqlQuery query(_db); query.prepare("SELECT tile_data FROM tiles " "WHERE zoom_level=:zoom AND tile_column=:x AND tile_row=:y"); query.bindValue(":zoom", zoom); query.bindValue(":x", tile.x()); query.bindValue(":y", (1< tiles; for (int i = 0; i < width; i++) { for (int j = 0; j < height; j++) { QPixmap pm; QPoint t(tile.x() + i, tile.y() + j); QString key = _fileName + "-" + QString::number(_zoom) + "_" + QString::number(t.x()) + "_" + QString::number(t.y()); if (QPixmapCache::find(key, pm)) { QPointF tp(qMax(tl.x(), b.left()) + (t.x() - tile.x()) * tileSize(), qMax(tl.y(), b.top()) + (t.y() - tile.y()) * tileSize()); drawTile(painter, pm, tp); } else tiles.append(MBTile(_zoom, t, tileData(_zoom, t), key)); } } QFuture future = QtConcurrent::map(tiles, render); future.waitForFinished(); for (int i = 0; i < tiles.size(); i++) { const MBTile &mt = tiles.at(i); QPixmap pm(mt.pixmap()); if (pm.isNull()) continue; QPixmapCache::insert(mt.key(), pm); QPointF tp(qMax(tl.x(), b.left()) + (mt.xy().x() - tile.x()) * tileSize(), qMax(tl.y(), b.top()) + (mt.xy().y() - tile.y()) * tileSize()); drawTile(painter, pm, tp); } } void MBTilesMap::drawTile(QPainter *painter, QPixmap &pixmap, QPointF &tp) { #ifdef ENABLE_HIDPI pixmap.setDevicePixelRatio(imageRatio()); #endif // ENABLE_HIDPI painter->drawPixmap(tp, pixmap); } QPointF MBTilesMap::ll2xy(const Coordinates &c) { qreal scale = OSM::zoom2scale(_zoom, _tileSize); QPointF m = OSM::ll2m(c); return QPointF(m.x() / scale, m.y() / -scale) / coordinatesRatio(); } Coordinates MBTilesMap::xy2ll(const QPointF &p) { qreal scale = OSM::zoom2scale(_zoom, _tileSize); return OSM::m2ll(QPointF(p.x() * scale, -p.y() * scale) * coordinatesRatio()); }