/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2014 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 * *****************************************************************************/ // mapnik #include #include #include #include #include #include #include #include #include #include #include // boost #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wunused-parameter" #pragma GCC diagnostic ignored "-Wunused-local-typedef" #include #include #pragma GCC diagnostic pop // agg #include "agg_rendering_buffer.h" #include "agg_pixfmt_rgba.h" namespace mapnik { marker_cache::marker_cache() : known_svg_prefix_("shape://"), known_image_prefix_("image://") { insert_svg("ellipse", "" "" "" ""); insert_svg("arrow", "" "" "" ""); boost::optional bitmap_data = boost::optional(std::make_shared(4,4)); (*bitmap_data)->set(0xff000000); marker_ptr mark = std::make_shared(bitmap_data); marker_cache_.emplace("image://square",mark); } marker_cache::~marker_cache() {} void marker_cache::clear() { #ifdef MAPNIK_THREADSAFE mapnik::scoped_lock lock(mutex_); #endif using iterator_type = boost::unordered_map::const_iterator; iterator_type itr = marker_cache_.begin(); while(itr != marker_cache_.end()) { if (!is_uri(itr->first)) { marker_cache_.erase(itr++); } else { ++itr; } } } bool marker_cache::is_svg_uri(std::string const& path) { return boost::algorithm::starts_with(path,known_svg_prefix_); } bool marker_cache::is_image_uri(std::string const& path) { return boost::algorithm::starts_with(path,known_image_prefix_); } bool marker_cache::insert_svg(std::string const& name, std::string const& svg_string) { std::string key = known_svg_prefix_ + name; using iterator_type = boost::unordered_map::const_iterator; iterator_type itr = svg_cache_.find(key); if (itr == svg_cache_.end()) { return svg_cache_.emplace(key,svg_string).second; } return false; } bool marker_cache::insert_marker(std::string const& uri, marker_ptr path) { #ifdef MAPNIK_THREADSAFE mapnik::scoped_lock lock(mutex_); #endif return marker_cache_.emplace(uri,path).second; } boost::optional marker_cache::find(std::string const& uri, bool update_cache) { boost::optional result; if (uri.empty()) { return result; } #ifdef MAPNIK_THREADSAFE mapnik::scoped_lock lock(mutex_); #endif using iterator_type = boost::unordered_map::const_iterator; iterator_type itr = marker_cache_.find(uri); if (itr != marker_cache_.end()) { result.reset(itr->second); return result; } try { // if uri references a built-in marker if (is_svg_uri(uri)) { boost::unordered_map::const_iterator mark_itr = svg_cache_.find(uri); if (mark_itr == svg_cache_.end()) { MAPNIK_LOG_ERROR(marker_cache) << "Marker does not exist: " << uri; return result; } std::string known_svg_string = mark_itr->second; using namespace mapnik::svg; svg_path_ptr marker_path(std::make_shared()); vertex_stl_adapter stl_storage(marker_path->source()); svg_path_adapter svg_path(stl_storage); svg_converter_type svg(svg_path, marker_path->attributes()); svg_parser p(svg); p.parse_from_string(known_svg_string); //svg.arrange_orientations(); double lox,loy,hix,hiy; svg.bounding_rect(&lox, &loy, &hix, &hiy); marker_path->set_bounding_box(lox,loy,hix,hiy); marker_path->set_dimensions(svg.width(),svg.height()); marker_ptr mark(std::make_shared(marker_path)); result.reset(mark); if (update_cache) { marker_cache_.emplace(uri,*result); } } // otherwise assume file-based else { if (!mapnik::util::exists(uri)) { MAPNIK_LOG_ERROR(marker_cache) << "Marker does not exist: " << uri; return result; } if (is_svg(uri)) { using namespace mapnik::svg; svg_path_ptr marker_path(std::make_shared()); vertex_stl_adapter stl_storage(marker_path->source()); svg_path_adapter svg_path(stl_storage); svg_converter_type svg(svg_path, marker_path->attributes()); svg_parser p(svg); p.parse(uri); //svg.arrange_orientations(); double lox,loy,hix,hiy; svg.bounding_rect(&lox, &loy, &hix, &hiy); marker_path->set_bounding_box(lox,loy,hix,hiy); marker_path->set_dimensions(svg.width(),svg.height()); marker_ptr mark(std::make_shared(marker_path)); result.reset(mark); if (update_cache) { marker_cache_.emplace(uri,*result); } } else { // TODO - support reading images from string std::unique_ptr reader(mapnik::get_image_reader(uri)); if (reader.get()) { unsigned width = reader->width(); unsigned height = reader->height(); BOOST_ASSERT(width > 0 && height > 0); mapnik::image_ptr image(std::make_shared(width,height)); reader->read(0,0,*image); if (!reader->premultiplied_alpha()) { agg::rendering_buffer buffer(image->getBytes(),image->width(),image->height(),image->width() * 4); agg::pixfmt_rgba32 pixf(buffer); pixf.premultiply(); } marker_ptr mark(std::make_shared(image)); result.reset(mark); if (update_cache) { marker_cache_.emplace(uri,*result); } } else { MAPNIK_LOG_ERROR(marker_cache) << "could not intialize reader for: '" << uri << "'"; } } } } catch (std::exception const& ex) { MAPNIK_LOG_ERROR(marker_cache) << "Exception caught while loading: '" << uri << "' (" << ex.what() << ")"; } return result; } }