#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/wgs84.h" #include "IMG/textpathitem.h" #include "IMG/textpointitem.h" #include "IMG/bitmapline.h" #include "pcs.h" #include "rectd.h" #include "imgmap.h" #define TILE_SIZE 256 #define TEXT_EXTENT 256 #define LARGE_FONT_SIZE 14 #define NORMAL_FONT_SIZE 12 #define SMALL_FONT_SIZE 10 #define POI_FONT_SIZE 9 #define LINE_TEXT_MIN_ZOOM 22 class RasterTile { public: RasterTile() : _map(0) {} RasterTile(IMGMap *map, const QPoint &xy, const QString &key) : _map(map), _xy(xy), _key(key), _img(TILE_SIZE, TILE_SIZE, QImage::Format_ARGB32_Premultiplied) {} const QString &key() const {return _key;} const QPoint &xy() const {return _xy;} QImage &img() {return _img;} QList &polygons() {return _polygons;} QList &lines() {return _lines;} QList &points() {return _points;} void load() { QList textItems; _map->processPolygons(_polygons); _map->processPoints(_points, textItems); _map->processLines(_lines, _xy, textItems); _img.fill(Qt::transparent); QPainter painter(&_img); painter.setRenderHint(QPainter::SmoothPixmapTransform); painter.setRenderHint(QPainter::Antialiasing); painter.translate(-_xy.x(), -_xy.y()); _map->drawPolygons(&painter, _polygons); _map->drawLines(&painter, _lines); _map->drawTextItems(&painter, textItems); qDeleteAll(textItems); } private: IMGMap *_map; QPoint _xy; QString _key; QImage _img; QList _polygons; QList _lines; QList _points; }; static void convertUnits(QString &str) { bool ok; int number = str.toInt(&ok); if (ok) str = QString::number(qRound(number * 0.3048)); } static int minPOIZoom(Style::POIClass cl) { switch (cl) { case Style::Food: case Style::Shopping: case Style::Services: return 27; case Style::Accommodation: case Style::Recreation: return 25; case Style::ManmadePlaces: case Style::NaturePlaces: case Style::Transport: case Style::Community: case Style::Elementary: return 23; default: return 0; } } static QFont font(int pixelSize) { QFont f; f.setPixelSize(pixelSize); return f; } /* The fonts must be initialized on first usage (after the QGuiApplication instance is created) */ #define FONT(name, size) \ static const QFont *name() \ { \ static QFont f = font(size); \ return &f; \ } FONT(largeFont, LARGE_FONT_SIZE) FONT(normalFont, NORMAL_FONT_SIZE) FONT(smallFont, SMALL_FONT_SIZE) FONT(poiFont, POI_FONT_SIZE) IMGMap::IMGMap(const QString &fileName, QObject *parent) : Map(parent), _fileName(fileName), _img(fileName), _projection(PCS::pcs(3857)), _valid(false) { if (!_img.isValid()) { _errorString = _img.errorString(); return; } _zooms = Range(_img.zooms().min() - 1, 28); _zoom = _zooms.min(); updateTransform(); _valid = true; } QRectF IMGMap::bounds() { RectD prect(_img.bounds(), _projection); return QRectF(_transform.proj2img(prect.topLeft()), _transform.proj2img(prect.bottomRight())); } int IMGMap::zoomFit(const QSize &size, const RectC &rect) { if (rect.isValid()) { RectD pr(rect, _projection, 10); _zoom = _zooms.min(); for (int i = _zooms.min() + 1; i <= _zooms.max(); i++) { Transform t(transform(i)); QRectF r(t.proj2img(pr.topLeft()), t.proj2img(pr.bottomRight())); if (size.width() < r.width() || size.height() < r.height()) break; _zoom = i; } } else _zoom = _zooms.max(); updateTransform(); return _zoom; } int IMGMap::zoomIn() { _zoom = qMin(_zoom + 1, _zooms.max()); updateTransform(); return _zoom; } int IMGMap::zoomOut() { _zoom = qMax(_zoom - 1, _zooms.min()); updateTransform(); return _zoom; } void IMGMap::setZoom(int zoom) { _zoom = zoom; updateTransform(); } Transform IMGMap::transform(int zoom) const { double scale = (2.0 * M_PI * WGS84_RADIUS) / (1< &polygons) { for (int n = 0; n < _img.style().drawOrder().size(); n++) { for (int i = 0; i < polygons.size(); i++) { const IMG::Poly &poly = polygons.at(i); if (poly.type != _img.style().drawOrder().at(n)) continue; const Style::Polygon &style = _img.style().polygon(poly.type); painter->setPen(style.pen()); painter->setBrush(style.brush()); painter->drawPolygon(poly.points); } } } void IMGMap::drawLines(QPainter *painter, const QList &lines) { painter->setBrush(Qt::NoBrush); for (int i = 0; i < lines.size(); i++) { const IMG::Poly &poly = lines.at(i); const Style::Line &style = _img.style().line(poly.type); if (style.background() == Qt::NoPen) continue; painter->setPen(style.background()); painter->drawPolyline(poly.points); } for (int i = 0; i < lines.size(); i++) { const IMG::Poly &poly = lines.at(i); const Style::Line &style = _img.style().line(poly.type); if (!style.img().isNull()) BitmapLine::draw(painter, poly.points, style.img()); else if (style.foreground() != Qt::NoPen) { painter->setPen(style.foreground()); painter->drawPolyline(poly.points); } } } void IMGMap::drawTextItems(QPainter *painter, const QList &textItems) { for (int i = 0; i < textItems.size(); i++) textItems.at(i)->paint(painter); } void IMGMap::processPolygons(QList &polygons) { for (int i = 0; i < polygons.size(); i++) { IMG::Poly &poly = polygons[i]; for (int j = 0; j < poly.points.size(); j++) { QPointF &p = poly.points[j]; p = ll2xy(Coordinates(p.x(), p.y())); } } } void IMGMap::processLines(QList &lines, const QPoint &tile, QList &textItems) { qStableSort(lines); for (int i = 0; i < lines.size(); i++) { IMG::Poly &poly = lines[i]; for (int j = 0; j < poly.points.size(); j++) { QPointF &p = poly.points[j]; p = ll2xy(Coordinates(p.x(), p.y())); } } if (_zoom < LINE_TEXT_MIN_ZOOM) return; for (int i = 0; i < lines.size(); i++) { IMG::Poly &poly = lines[i]; const Style::Line &style = _img.style().line(poly.type); if (style.img().isNull() && style.foreground() == Qt::NoPen) continue; if (poly.label.isEmpty() || style.textFontSize() == Style::None) continue; if (Style::isContourLine(poly.type)) convertUnits(poly.label); const QFont *font; switch (style.textFontSize()) { case Style::Large: font = largeFont(); break; case Style::Small: font = smallFont(); break; default: font = normalFont(); } const QColor *color = style.textColor().isValid() ? &style.textColor() : 0; TextPathItem *item = new TextPathItem(poly.points, &poly.label, QRect(tile, QSize(TILE_SIZE, TILE_SIZE)), font, color); if (item->isValid() && !item->collides(textItems)) textItems.append(item); else delete item; } } void IMGMap::processPoints(QList &points, QList &textItems) { qSort(points); for (int i = 0; i < points.size(); i++) { IMG::Point &point = points[i]; const Style::Point &style = _img.style().point(point.type); if (point.poi && _zoom < minPOIZoom(Style::poiClass(point.type))) continue; const QString *label = point.label.isEmpty() ? 0 : &(point.label); const QImage *img = style.img().isNull() ? 0 : &style.img(); const QFont *font = 0; if (point.poi) { if (style.textFontSize() == Style::None) label = 0; else font = poiFont(); } else { switch (style.textFontSize()) { case Style::None: label = 0; break; case Style::Normal: font = normalFont(); break; case Style::Small: font = smallFont(); break; default: font = largeFont(); } } const QColor *color = style.textColor().isValid() ? &style.textColor() : 0; if (!label && !img) continue; if (Style::isSpot(point.type)) convertUnits(point.label); if (Style::isSummit(point.type) && !point.label.isEmpty()) { QStringList list = point.label.split(" "); convertUnits(list.last()); point.label = list.join(" "); } TextPointItem *item = new TextPointItem( ll2xy(point.coordinates).toPoint(), label, font, img, color); if (item->isValid() && !item->collides(textItems)) textItems.append(item); else delete item; } } static void render(RasterTile &tile) { tile.load(); } void IMGMap::draw(QPainter *painter, const QRectF &rect, Flags flags) { Q_UNUSED(flags); QPointF tl(floor(rect.left() / TILE_SIZE) * TILE_SIZE, floor(rect.top() / TILE_SIZE) * TILE_SIZE); QSizeF s(rect.right() - tl.x(), rect.bottom() - tl.y()); int width = ceil(s.width() / TILE_SIZE); int height = ceil(s.height() / TILE_SIZE); QList tiles; for (int i = 0; i < width; i++) { for (int j = 0; j < height; j++) { QPixmap pm; QPoint ttl(tl.x() + i * TILE_SIZE, tl.y() + j * TILE_SIZE); QString key = _fileName + "-" + QString::number(_zoom) + "_" + QString::number(ttl.x()) + "_" + QString::number(ttl.y()); if (QPixmapCache::find(key, pm)) painter->drawPixmap(ttl, pm); else { tiles.append(RasterTile(this, ttl, key)); RasterTile &tile = tiles.last(); RectD polyRect(_transform.img2proj(ttl), _transform.img2proj( QPointF(ttl.x() + TILE_SIZE, ttl.y() + TILE_SIZE))); _img.objects(polyRect.toRectC(_projection, 4), _zoom, &(tile.polygons()), &(tile.lines()), 0); RectD pointRect(_transform.img2proj(QPointF(ttl.x() - TEXT_EXTENT, ttl.y() - TEXT_EXTENT)), _transform.img2proj(QPointF(ttl.x() + TILE_SIZE + TEXT_EXTENT, ttl.y() + TILE_SIZE + TEXT_EXTENT))); _img.objects(pointRect.toRectC(_projection, 4), _zoom, 0, 0, &(tile.points())); } } } QFuture future = QtConcurrent::map(tiles, render); future.waitForFinished(); for (int i = 0; i < tiles.size(); i++) { RasterTile &mt = tiles[i]; QPixmap pm(QPixmap::fromImage(mt.img())); if (pm.isNull()) continue; QPixmapCache::insert(mt.key(), pm); painter->drawPixmap(mt.xy(), pm); } } void IMGMap::setProjection(const Projection &projection) { _projection = projection; updateTransform(); QPixmapCache::clear(); }