/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2012 Artem Pavlenko * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * *****************************************************************************/ #include "geojson_datasource.hpp" #include "geojson_featureset.hpp" #include #include // boost #include #include #include #include #include #include #include #include // mapnik #include #include #include #include using mapnik::datasource; using mapnik::parameters; DATASOURCE_PLUGIN(geojson_datasource) geojson_datasource::geojson_datasource(parameters const& params, bool bind) : datasource(params), type_(datasource::Vector), desc_(*params_.get("type"), *params_.get("encoding","utf-8")), file_(*params_.get("file","")), extent_(), tr_(new mapnik::transcoder(*params_.get("encoding","utf-8"))), features_(), tree_(16,1) { if (file_.empty()) throw mapnik::datasource_exception("GeoJSON Plugin: missing parameter"); if (bind) { this->bind(); } } void geojson_datasource::bind() const { if (is_bound_) return; typedef std::istreambuf_iterator base_iterator_type; std::ifstream is(file_.c_str()); boost::spirit::multi_pass begin = boost::spirit::make_default_multi_pass(base_iterator_type(is)); boost::spirit::multi_pass end = boost::spirit::make_default_multi_pass(base_iterator_type()); mapnik::context_ptr ctx = boost::make_shared(); mapnik::json::feature_collection_parser > p(ctx,*tr_); bool result = p.parse(begin,end, features_); if (!result) { MAPNIK_LOG_WARN(geojson) << "geojson_datasource: Failed parse GeoJSON file " << file_; return; } bool first = true; std::size_t count=0; BOOST_FOREACH (mapnik::feature_ptr f, features_) { mapnik::box2d const& box = f->envelope(); if (first) { extent_ = box; first = false; } else { extent_.expand_to_include(box); } tree_.insert(box_type(point_type(box.minx(),box.miny()),point_type(box.maxx(),box.maxy())), count++); } is_bound_ = true; } geojson_datasource::~geojson_datasource() { } const char * geojson_datasource::name() { return "geojson"; } boost::optional geojson_datasource::get_geometry_type() const { return boost::optional(); } mapnik::datasource::datasource_t geojson_datasource::type() const { return type_; } std::map geojson_datasource::get_statistics() const { return statistics_; } mapnik::box2d geojson_datasource::envelope() const { if (!is_bound_) bind(); return extent_; } mapnik::layer_descriptor geojson_datasource::get_descriptor() const { if (!is_bound_) bind(); return desc_; } mapnik::featureset_ptr geojson_datasource::features(mapnik::query const& q) const { if (!is_bound_) bind(); // if the query box intersects our world extent then query for features mapnik::box2d const& b = q.get_bbox(); if (extent_.intersects(b)) { box_type box(point_type(b.minx(),b.miny()),point_type(b.maxx(),b.maxy())); index_array_ = tree_.find(box); return boost::make_shared(features_, index_array_.begin(), index_array_.end()); } // otherwise return an empty featureset pointer return mapnik::featureset_ptr(); } // FIXME mapnik::featureset_ptr geojson_datasource::features_at_point(mapnik::coord2d const& pt) const { if (!is_bound_) bind(); return mapnik::featureset_ptr(); }