mapnik/demo/viewer/mapwidget.cpp

624 lines
16 KiB
C++
Raw Normal View History

2007-08-07 17:09:41 +02:00
/* This file is part of Mapnik (c++ mapping toolkit)
2011-10-23 16:09:12 +02:00
*
* Copyright (C) 2011 Artem Pavlenko
2007-08-07 17:09:41 +02:00
*
* Mapnik is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include <QtGui>
2013-02-27 15:59:53 +01:00
#define BOOST_CHRONO_HEADER_ONLY
2013-09-20 03:30:28 +02:00
#include <boost/chrono/process_cpu_clocks.hpp>
#include <boost/chrono.hpp>
2007-08-07 17:09:41 +02:00
#include <boost/bind.hpp>
#include <mapnik/agg_renderer.hpp>
#include <mapnik/graphics.hpp>
2007-08-07 17:09:41 +02:00
#include <mapnik/layer.hpp>
#include <mapnik/projection.hpp>
2011-09-08 10:21:41 +02:00
#include <mapnik/scale_denominator.hpp>
2007-08-07 17:09:41 +02:00
#include <mapnik/ctrans.hpp>
#include <mapnik/memory_datasource.hpp>
#include <mapnik/feature_kv_iterator.hpp>
#include <mapnik/config_error.hpp>
#include <mapnik/image_util.hpp>
#ifdef HAVE_CAIRO
// cairo
#include <mapnik/cairo_renderer.hpp>
#endif
2007-08-07 17:09:41 +02:00
#include "mapwidget.hpp"
#include "info_dialog.hpp"
2009-12-16 21:02:06 +01:00
using mapnik::image_32;
2007-08-07 17:09:41 +02:00
using mapnik::Map;
2009-12-16 21:02:06 +01:00
using mapnik::layer;
using mapnik::box2d;
2007-08-07 17:09:41 +02:00
using mapnik::coord2d;
using mapnik::feature_ptr;
using mapnik::geometry_ptr;
using mapnik::CoordTransform;
using mapnik::projection;
2011-09-08 10:21:41 +02:00
using mapnik::scale_denominator;
using mapnik::feature_kv_iterator;
2007-08-07 17:09:41 +02:00
double scales [] = {279541132.014,
139770566.007,
69885283.0036,
34942641.5018,
17471320.7509,
8735660.37545,
4367830.18772,
2183915.09386,
1091957.54693,
545978.773466,
272989.386733,
136494.693366,
68247.3466832,
34123.6733416,
17061.8366708,
8530.9183354,
4265.4591677,
2132.72958385,
1066.36479192,
533.182395962};
2011-10-23 16:09:12 +02:00
MapWidget::MapWidget(QWidget *parent)
: QWidget(parent),
map_(),
selected_(1),
extent_(),
cur_tool_(ZoomToBox),
start_x_(0),
start_y_(0),
end_x_(0),
end_y_(0),
drag_(false),
first_(true),
pen_(QColor(0,0,255,96)),
selectedLayer_(-1),
scaling_factor_(1.0),
cur_renderer_(AGG)
2007-08-07 17:09:41 +02:00
{
pen_.setWidth(3);
pen_.setCapStyle(Qt::RoundCap);
pen_.setJoinStyle(Qt::RoundJoin);
}
void MapWidget::setTool(eTool tool)
{
cur_tool_=tool;
}
void MapWidget::paintEvent(QPaintEvent*)
2011-10-23 16:09:12 +02:00
{
2007-08-07 17:09:41 +02:00
QPainter painter(this);
2011-10-23 16:09:12 +02:00
2007-08-07 17:09:41 +02:00
if (drag_)
{
if (cur_tool_ == ZoomToBox)
{
unsigned width = end_x_-start_x_;
unsigned height = end_y_-start_y_;
painter.drawPixmap(QPoint(0, 0),pix_);
painter.setPen(pen_);
painter.setBrush(QColor(200,200,255,128));
2007-08-07 17:09:41 +02:00
painter.drawRect(start_x_,start_y_,width,height);
}
else if (cur_tool_ == Pan)
{
int dx = end_x_-start_x_;
int dy = end_y_-start_y_;
2010-06-01 15:30:53 +02:00
painter.setBrush(QColor(200,200,200,128));
2007-08-07 17:09:41 +02:00
painter.drawRect(0,0,width(),height());
painter.drawPixmap(QPoint(dx,dy),pix_);
}
}
else
{
painter.drawPixmap(QPoint(0, 0),pix_);
}
painter.end();
}
2011-10-23 16:09:12 +02:00
2007-08-07 17:09:41 +02:00
void MapWidget::resizeEvent(QResizeEvent * ev)
{
if (map_)
{
map_->resize(ev->size().width(),ev->size().height());
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
2011-10-23 16:09:12 +02:00
2007-08-07 17:09:41 +02:00
void MapWidget::mousePressEvent(QMouseEvent* e)
{
2011-10-23 16:09:12 +02:00
if (e->button()==Qt::LeftButton)
2007-08-07 17:09:41 +02:00
{
if (cur_tool_ == ZoomToBox || cur_tool_==Pan)
{
start_x_ = e->x();
start_y_ = e->y();
drag_=true;
}
else if (cur_tool_==Info)
{
if (map_)
2011-10-23 16:09:12 +02:00
{
2007-08-07 17:09:41 +02:00
QVector<QPair<QString,QString> > info;
2011-10-23 16:09:12 +02:00
projection map_proj(map_->srs()); // map projection
2013-02-21 11:39:02 +01:00
double scale_denom = scale_denominator(map_->scale(),map_proj.is_geographic());
2010-06-25 17:23:35 +02:00
CoordTransform t(map_->width(),map_->height(),map_->get_current_extent());
2011-10-23 16:09:12 +02:00
2010-06-25 17:23:35 +02:00
for (unsigned index = 0; index < map_->layer_count();++index)
2007-08-07 17:09:41 +02:00
{
if (int(index) != selectedLayer_) continue;
2011-10-23 16:09:12 +02:00
2009-12-16 21:02:06 +01:00
layer & layer = map_->layers()[index];
if (!layer.visible(scale_denom)) continue;
2007-08-07 17:09:41 +02:00
std::string name = layer.name();
double x = e->x();
double y = e->y();
std::cout << "query at " << x << "," << y << "\n";
projection layer_proj(layer.srs());
mapnik::proj_transform prj_trans(map_proj,layer_proj);
//std::auto_ptr<mapnik::memory_datasource> data(new mapnik::memory_datasource);
2007-08-07 17:09:41 +02:00
mapnik::featureset_ptr fs = map_->query_map_point(index,x,y);
2011-10-23 16:09:12 +02:00
2007-08-07 17:09:41 +02:00
if (fs)
{
feature_ptr feat = fs->next();
2011-10-23 16:09:12 +02:00
if (feat)
2007-08-07 17:09:41 +02:00
{
feature_kv_iterator itr(*feat,true);
feature_kv_iterator end(*feat);
for ( ;itr!=end; ++itr)
{
info.push_back(QPair<QString,QString>(QString(std::get<0>(*itr).c_str()),
std::get<1>(*itr).to_string().c_str()));
}
typedef mapnik::coord_transform<mapnik::CoordTransform,mapnik::geometry_type> path_type;
2011-10-23 16:09:12 +02:00
for (unsigned i=0; i<feat->num_geometries();++i)
2007-08-07 17:09:41 +02:00
{
2011-10-23 16:09:12 +02:00
mapnik::geometry_type & geom = feat->get_geometry(i);
path_type path(t,geom,prj_trans);
if (geom.size() > 0)
{
QPainterPath qpath;
double x,y;
path.vertex(&x,&y);
qpath.moveTo(x,y);
for (unsigned j = 1; j < geom.size(); ++j)
{
path.vertex(&x,&y);
qpath.lineTo(x,y);
}
QPainter painter(&pix_);
QPen pen(QColor(255,0,0,96));
pen.setWidth(3);
pen.setCapStyle(Qt::RoundCap);
pen.setJoinStyle(Qt::RoundJoin);
painter.setPen(pen);
painter.drawPath(qpath);
update();
}
2007-08-07 17:09:41 +02:00
}
}
}
2011-10-23 16:09:12 +02:00
if (info.size() > 0)
2007-08-07 17:09:41 +02:00
{
info_dialog info_dlg(info,this);
info_dlg.exec();
break;
}
}
2011-10-23 16:09:12 +02:00
2007-08-07 17:09:41 +02:00
// remove annotation layer
map_->layers().erase(remove_if(map_->layers().begin(),
map_->layers().end(),
2009-12-16 21:02:06 +01:00
bind(&layer::name,_1) == "*annotations*")
2007-08-07 17:09:41 +02:00
, map_->layers().end());
}
2011-10-23 16:09:12 +02:00
}
2007-08-07 17:09:41 +02:00
}
2011-10-23 16:09:12 +02:00
else if (e->button()==Qt::RightButton)
{
2008-06-29 12:58:29 +02:00
//updateMap();
2007-08-07 17:09:41 +02:00
}
}
2011-10-23 16:09:12 +02:00
2007-08-07 17:09:41 +02:00
void MapWidget::mouseMoveEvent(QMouseEvent* e)
2011-10-23 16:09:12 +02:00
{
2007-08-07 17:09:41 +02:00
if (cur_tool_ == ZoomToBox || cur_tool_==Pan)
{
end_x_ = e->x();
end_y_ = e->y();
update();
2011-10-23 16:09:12 +02:00
}
2007-08-07 17:09:41 +02:00
}
void MapWidget::mouseReleaseEvent(QMouseEvent* e)
{
2011-10-23 16:09:12 +02:00
if (e->button()==Qt::LeftButton)
2007-08-07 17:09:41 +02:00
{
end_x_ = e->x();
end_y_ = e->y();
if (cur_tool_ == ZoomToBox)
{
drag_=false;
if (map_)
{
2011-10-23 16:09:12 +02:00
CoordTransform t(map_->width(),map_->height(),map_->get_current_extent());
2009-12-16 21:02:06 +01:00
box2d<double> box = t.backward(box2d<double>(start_x_,start_y_,end_x_,end_y_));
2010-06-25 17:23:35 +02:00
map_->zoom_to_box(box);
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
else if (cur_tool_==Pan)
{
drag_=false;
if (map_)
{
2010-06-25 17:23:35 +02:00
int cx = int(0.5 * map_->width());
int cy = int(0.5 * map_->height());
2007-08-07 17:09:41 +02:00
int dx = end_x_ - start_x_;
int dy = end_y_ - start_y_;
2011-10-23 16:09:12 +02:00
map_->pan(cx - dx ,cy - dy);
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
}
}
void MapWidget::wheelEvent(QWheelEvent* e)
{
if (!map_)
{
return;
}
QPoint corner(map_->width(), map_->height());
QPoint zoomCoords;
double zoom;
if (e->delta() > 0)
{
zoom = 0.5;
QPoint center = corner / 2;
QPoint delta = e->pos() - center;
zoomCoords = zoom * delta + center;
}
else
{
zoom = 2.0;
zoomCoords = corner - e->pos();
}
map_->pan_and_zoom(zoomCoords.x(), zoomCoords.y(), zoom);
updateMap();
}
2007-08-07 17:09:41 +02:00
2011-10-23 16:09:12 +02:00
void MapWidget::keyPressEvent(QKeyEvent *e)
2007-08-07 17:09:41 +02:00
{
2009-12-16 21:02:06 +01:00
std::cout << "key pressed:"<< e->key()<<"\n";
2011-10-23 16:09:12 +02:00
switch (e->key()) {
2007-08-07 17:09:41 +02:00
case Qt::Key_Minus:
zoomOut();
break;
case Qt::Key_Plus:
case 61:
zoomIn();
break;
case 65:
defaultView();
break;
case Qt::Key_Up:
panUp();
break;
case Qt::Key_Down:
panDown();
2011-10-23 16:09:12 +02:00
break;
2007-08-07 17:09:41 +02:00
case Qt::Key_Left:
panLeft();
2011-10-23 16:09:12 +02:00
break;
2007-08-07 17:09:41 +02:00
case Qt::Key_Right:
panRight();
break;
case 49:
zoomToLevel(10);
break;
case 50:
zoomToLevel(11);
break;
case 51:
zoomToLevel(12);
2011-10-23 16:09:12 +02:00
break;
2007-08-07 17:09:41 +02:00
case 52:
zoomToLevel(13);
break;
case 53:
zoomToLevel(14);
break;
case 54:
zoomToLevel(15);
2011-10-23 16:09:12 +02:00
break;
2007-08-07 17:09:41 +02:00
case 55:
zoomToLevel(16);
break;
case 56:
zoomToLevel(17);
break;
case 57:
zoomToLevel(18);
2011-10-23 16:09:12 +02:00
break;
2009-12-16 21:02:06 +01:00
default:
QWidget::keyPressEvent(e);
2007-08-07 17:09:41 +02:00
}
2011-10-23 16:09:12 +02:00
2007-08-07 17:09:41 +02:00
}
2009-12-16 21:02:06 +01:00
void MapWidget::zoomToBox(mapnik::box2d<double> const& bbox)
2007-08-07 17:09:41 +02:00
{
if (map_)
{
2010-06-25 17:23:35 +02:00
map_->zoom_to_box(bbox);
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
2011-10-23 16:09:12 +02:00
void MapWidget::defaultView()
2007-08-07 17:09:41 +02:00
{
if (map_)
{
map_->resize(width(),height());
map_->zoom_all();
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
2011-10-23 16:09:12 +02:00
void MapWidget::zoomIn()
2007-08-07 17:09:41 +02:00
{
if (map_)
{
map_->zoom(0.5);
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
2011-10-23 16:09:12 +02:00
void MapWidget::zoomOut()
2007-08-07 17:09:41 +02:00
{
if (map_)
{
map_->zoom(2.0);
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
void MapWidget::panUp()
{
if (map_)
{
2010-06-25 17:23:35 +02:00
double cx = 0.5*map_->width();
double cy = 0.5*map_->height();
2007-08-07 17:09:41 +02:00
map_->pan(int(cx),int(cy - cy*0.25));
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
void MapWidget::panDown()
{
if (map_)
{
2010-06-25 17:23:35 +02:00
double cx = 0.5*map_->width();
double cy = 0.5*map_->height();
2007-08-07 17:09:41 +02:00
map_->pan(int(cx),int(cy + cy*0.25));
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
void MapWidget::panLeft()
{
if (map_)
{
2010-06-25 17:23:35 +02:00
double cx = 0.5*map_->width();
double cy = 0.5*map_->height();
2007-08-07 17:09:41 +02:00
map_->pan(int(cx - cx * 0.25),int(cy));
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
void MapWidget::panRight()
2011-10-23 16:09:12 +02:00
{
2007-08-07 17:09:41 +02:00
if (map_)
{
2010-06-25 17:23:35 +02:00
double cx = 0.5*map_->width();
double cy = 0.5*map_->height();
2011-10-23 16:09:12 +02:00
map_->pan(int(cx + cx * 0.25),int(cy));
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
void MapWidget::zoomToLevel(int level)
{
if ( map_ && level >= 0 && level < 19 )
{
double scale_denom = scales[level];
std::cerr << "scale denominator = " << scale_denom << "\n";
2010-06-25 17:23:35 +02:00
mapnik::box2d<double> ext = map_->get_current_extent();
double width = static_cast<double>(map_->width());
2011-10-23 16:09:12 +02:00
double height= static_cast<double>(map_->height());
2007-08-07 17:09:41 +02:00
mapnik::coord2d pt = ext.center();
double res = scale_denom * 0.00028;
2011-10-23 16:09:12 +02:00
2009-12-16 21:02:06 +01:00
mapnik::box2d<double> box(pt.x - 0.5 * width * res,
2007-08-07 17:09:41 +02:00
pt.y - 0.5 * height*res,
pt.x + 0.5 * width * res,
pt.y + 0.5 * height*res);
2010-06-25 17:23:35 +02:00
map_->zoom_to_box(box);
2008-06-29 12:58:29 +02:00
updateMap();
2007-08-07 17:09:41 +02:00
}
}
2011-10-23 16:09:12 +02:00
void MapWidget::export_to_file(unsigned ,unsigned ,std::string const&,std::string const&)
2007-08-07 17:09:41 +02:00
{
2009-12-16 21:02:06 +01:00
//image_32 image(width,height);
2007-08-07 17:09:41 +02:00
//agg_renderer renderer(map,image);
//renderer.apply();
//image.saveToFile(filename,type);
std::cout << "Export to file .." << std::endl;
2007-08-07 17:09:41 +02:00
}
2011-10-23 16:09:12 +02:00
void MapWidget::set_scaling_factor(double scaling_factor)
{
scaling_factor_ = scaling_factor;
}
void render_agg(mapnik::Map const& map, double scaling_factor, QPixmap & pix)
2011-10-23 16:09:12 +02:00
{
unsigned width=map.width();
unsigned height=map.height();
image_32 buf(width,height);
mapnik::agg_renderer<image_32> ren(map,buf,scaling_factor);
try
{
2013-02-27 15:59:53 +01:00
{
2013-09-20 03:30:28 +02:00
boost::chrono::process_cpu_clock::time_point start = boost::chrono::process_cpu_clock::now();
2013-02-27 15:59:53 +01:00
ren.apply();
2013-09-20 03:30:28 +02:00
boost::chrono::process_cpu_clock::duration elapsed = boost::chrono::process_cpu_clock::now() - start;
std::clog << "rendering took: " << boost::chrono::duration_cast<boost::chrono::milliseconds>(elapsed) << "\n";
2013-02-27 15:59:53 +01:00
}
QImage image((uchar*)buf.raw_data(),width,height,QImage::Format_ARGB32);
pix = QPixmap::fromImage(image.rgbSwapped());
}
catch (mapnik::config_error & ex)
{
std::cerr << ex.what() << std::endl;
}
catch (const std::exception & ex)
{
std::cerr << "exception: " << ex.what() << std::endl;
}
catch (...)
{
std::cerr << "Unknown exception caught!\n";
}
}
2011-10-23 16:09:12 +02:00
2009-12-21 14:45:07 +01:00
void render_grid(mapnik::Map const& map, double scaling_factor, QPixmap & pix)
{
2013-05-16 17:24:18 +02:00
std::cerr << "Not supported" << std::endl;
}
void render_cairo(mapnik::Map const& map, double scaling_factor, QPixmap & pix)
{
2011-10-23 16:09:12 +02:00
#ifdef HAVE_CAIRO
mapnik::cairo_surface_ptr image_surface(cairo_image_surface_create(CAIRO_FORMAT_ARGB32,map.width(),map.height()),
mapnik::cairo_surface_closer());
mapnik::cairo_renderer<mapnik::cairo_surface_ptr> renderer(map, image_surface, scaling_factor);
renderer.apply();
image_32 buf(image_surface);
QImage image((uchar*)buf.raw_data(),buf.width(),buf.height(),QImage::Format_ARGB32);
pix = QPixmap::fromImage(image.rgbSwapped());
#endif
}
2011-10-23 16:09:12 +02:00
void MapWidget::updateRenderer(QString const& txt)
{
if (txt == "AGG") cur_renderer_ = AGG;
else if (txt == "Cairo") cur_renderer_ = Cairo;
else if (txt == "Grid") cur_renderer_ = Grid;
std::cerr << "Update renderer called" << std::endl;
updateMap();
}
void MapWidget::updateScaleFactor(double scale_factor)
{
set_scaling_factor(scale_factor);
updateMap();
}
void MapWidget::updateMap()
{
if (map_)
{
if (cur_renderer_== AGG)
{
render_agg(*map_, scaling_factor_, pix_);
}
else if (cur_renderer_ == Cairo)
{
render_cairo(*map_, scaling_factor_, pix_);
}
else if (cur_renderer_ == Grid)
{
render_grid(*map_, scaling_factor_, pix_);
}
else
{
std::cerr << "Unknown renderer..." << std::endl;
}
try
{
projection prj(map_->srs()); // map projection
2010-06-25 17:23:35 +02:00
box2d<double> ext = map_->get_current_extent();
double x0 = ext.minx();
double y0 = ext.miny();
double x1 = ext.maxx();
double y1 = ext.maxy();
prj.inverse(x0,y0);
prj.inverse(x1,y1);
std::cout << "BBOX (WGS84): " << x0 << "," << y0 << "," << x1 << "," << y1 << "\n";
update();
// emit signal to interested widgets
emit mapViewChanged();
}
2009-12-21 14:45:07 +01:00
catch (...)
{
std::cerr << "Unknown exception caught!\n";
2009-12-21 14:45:07 +01:00
}
2008-06-29 12:58:29 +02:00
}
2007-08-07 17:09:41 +02:00
}
std::shared_ptr<Map> MapWidget::getMap()
2007-08-07 17:09:41 +02:00
{
return map_;
}
void MapWidget::setMap(std::shared_ptr<Map> map)
2007-08-07 17:09:41 +02:00
{
map_ = map;
}
void MapWidget::layerSelected(int index)
{
selectedLayer_ = index;
}