#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 23 #define POI_MIN_ZOOM 25 #define POI_TEXT_MIN_ZOOM 26 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() { _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, _xy); _map->drawPoints(&painter, _points); } 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)); } 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(); _largeFont.setPixelSize(LARGE_FONT_SIZE); _normalFont.setPixelSize(NORMAL_FONT_SIZE); _smallFont.setPixelSize(SMALL_FONT_SIZE); _poiFont.setPixelSize(POI_FONT_SIZE); _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()) { QPointF sc((rect.right() - rect.left()) / size.width(), (rect.top() - rect.bottom()) / size.height()); double resolution = qMax(qAbs(sc.x()), qAbs(sc.y())); _zoom = _zooms.min(); for (int i = _zooms.min(); i <= _zooms.max(); i++) { if (360.0 / (1< &polygons) { for (int n = 0; n < _img.style().drawOrder().size(); n++) { for (int i = 0; i < polygons.size(); i++) { IMG::Poly &poly = polygons[i]; if (poly.type != _img.style().drawOrder().at(n)) continue; const Style::Polygon &style = _img.style().polygon(poly.type); for (int j = 0; j < poly.points.size(); j++) { QPointF &p = poly.points[j]; p = ll2xy(Coordinates(p.x(), p.y())); } painter->setPen(style.pen()); painter->setBrush(style.brush()); painter->drawPolygon(poly.points); } } } void IMGMap::drawLines(QPainter *painter, QList &lines, const QPoint &tile) { 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())); } } 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); } } if (_zoom < LINE_TEXT_MIN_ZOOM) return; QVector items; 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(poly.points, &poly.label, QRect(tile, QSize(TILE_SIZE, TILE_SIZE)), font, color); if (item.isValid() && !item.collides(items)) items.append(item); } for (int i = 0; i < items.size(); i++) items.at(i).paint(painter); } void IMGMap::drawPoints(QPainter *painter, QList &points) { qSort(points); QVector items; for (int i = 0; i < points.size(); i++) { const IMG::Point &point = points.at(i); const Style::Point &style = _img.style().point(point.type); if (point.poi && _zoom < POI_MIN_ZOOM) continue; const QString *label = ((point.poi && _zoom < POI_TEXT_MIN_ZOOM) || point.label.isEmpty()) ? 0 : &(point.label); const QImage *img = style.img().isNull() ? 0 : &style.img(); const QFont *font = 0; if (point.poi) 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; TextPointItem item(ll2xy(point.coordinates).toPoint(), label, font, img, color); if (!item.collides(items)) items.append(item); } for (int i = 0; i < items.size(); i++) items.at(i).paint(painter); } 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); } }