1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
|
/*
* Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved.
*
* This program 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; version 2 of the
* License.
*
* 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., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA
*/
#include "geom_draw_box.h"
#include "base/log.h"
#include "grt/spatial_handler.h"
DEFAULT_LOG_DOMAIN("GeomDrawBox");
void GeomDrawBox::draw_ring(cairo_t *cr, OGRRawPoint *points, int num_points, double scale, double x, double y, double height)
{
cairo_move_to(cr, (points[0].x-x) * scale, height - (points[0].y-y) * scale);
for (int i = 1; i < num_points; i++)
{
cairo_line_to(cr, (points[i].x-x) * scale, height - (points[i].y-y) * scale);
}
}
void GeomDrawBox::draw_ring_vertices(cairo_t *cr, OGRRawPoint *points, int num_points, double scale, double x, double y, double height)
{
cairo_arc(cr, (points[0].x-x) * scale, height - (points[0].y-y) * scale, 2, 0, 2*M_PI);
cairo_fill(cr);
for (int i = 1; i < num_points; i++)
{
cairo_arc(cr, (points[i].x-x) * scale, height - (points[i].y-y) * scale, 2, 0, 2*M_PI);
cairo_fill(cr);
}
}
void GeomDrawBox::draw_polygon(cairo_t *cr, OGRPolygon *poly, double scale, double x, double y, double height)
{
const OGRLinearRing *ring = poly->getExteriorRing();
if (ring->getNumPoints() > 0)
{
OGRRawPoint *points = new OGRRawPoint[ring->getNumPoints()];
ring->getPoints(points);
draw_ring(cr, points, ring->getNumPoints(), scale, x, y, height);
cairo_set_line_width(cr, 1);
cairo_set_source_rgb(cr, 0, 0, 0);
cairo_stroke_preserve(cr);
cairo_set_source_rgb(cr, 0.8, 0.8, 0.8);
cairo_fill(cr);
cairo_set_source_rgb(cr, 1, 0, 0);
draw_ring_vertices(cr, points, ring->getNumPoints(), scale, x, y, height);
delete []points;
}
}
void GeomDrawBox::set_data(const std::string &text)
{
spatial::Importer importer;
importer.import_from_mysql(text);
_geom = importer.steal_data();
set_needs_repaint();
}
void GeomDrawBox::draw_geometry(cairo_t *cr, OGRGeometry *geom, double scale, double x, double y, double height)
{
switch (geom->getGeometryType())
{
case wkbPolygon:
draw_polygon(cr, dynamic_cast<OGRPolygon*>(geom), scale, x, y, height);
break;
case wkbMultiPolygon:
{
OGRGeometryCollection *geoCollection = dynamic_cast<OGRGeometryCollection*>(geom);
for (int i = 0; i < geoCollection->getNumGeometries(); ++i)
draw_geometry(cr, geoCollection->getGeometryRef(i), scale, x, y, height);
}
break;
default:
log_warning("Can't paint geometry type %s\n", geom->getGeometryName());
break;
}
}
void GeomDrawBox::repaint(cairo_t *cr, int x, int y, int w, int h)
{
if (_geom)
{
OGREnvelope env;
_geom->getEnvelope(&env);
double fig_width = env.MaxX - env.MinX;
double fig_height = env.MaxY - env.MinY;
double scale;
int padding = 5;
if (fig_width > fig_height)
scale = (get_width() - padding*2) / fig_width;
else
scale = (get_height() - padding*2) / fig_height;
cairo_translate(cr, padding, padding);
draw_geometry(cr, _geom, scale, env.MinX, env.MinY, get_height()-padding*2);
}
}
|