/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2013 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 "boost_std_shared_shim.hpp" // FIXME - do we need it? // The functions in this file produce deprecation warnings. // But as shield symbolizer doesn't fully support more than one // placement from python yet these functions are actually the // correct ones. #define NO_DEPRECATION_WARNINGS // boost #include #include #include // mapnik #include #include #include #include #include #include #include #include "mapnik_enumeration.hpp" #include "mapnik_svg.hpp" #include #include #include #include // for known_svg_prefix_ // stl #include using mapnik::symbolizer; using mapnik::point_symbolizer; using mapnik::line_symbolizer; using mapnik::line_pattern_symbolizer; using mapnik::polygon_symbolizer; using mapnik::polygon_pattern_symbolizer; using mapnik::raster_symbolizer; using mapnik::shield_symbolizer; using mapnik::text_symbolizer; using mapnik::building_symbolizer; using mapnik::markers_symbolizer; using mapnik::symbolizer_base; using mapnik::color; using mapnik::path_processor_type; using mapnik::path_expression_ptr; using mapnik::guess_type; using mapnik::expression_ptr; using mapnik::parse_path; namespace { using namespace boost::python; void __setitem__(mapnik::symbolizer_base & sym, std::string const& name, mapnik::symbolizer_base::value_type const& val) { //std::cerr << "__setitem__ " << typeid(val).name() << std::endl; put(sym, mapnik::get_key(name), val); } struct extract_python_object : public boost::static_visitor { typedef boost::python::object result_type; template auto operator() (T const& val) const -> result_type { return result_type(val); // wrap into python object } }; boost::python::object __getitem__(mapnik::symbolizer_base const& sym, std::string const& name) { typedef symbolizer_base::cont_type::const_iterator const_iterator; mapnik::keys key = mapnik::get_key(name); const_iterator itr = sym.properties.find(key); if (itr != sym.properties.end()) { return boost::apply_visitor(extract_python_object(), itr->second); } //mapnik::property_meta_type const& meta = mapnik::get_meta(key); //return boost::apply_visitor(extract_python_object(), std::get<1>(meta)); return boost::python::object(); } struct symbolizer_to_json : public boost::static_visitor { typedef std::string result_type; template auto operator() (T const& sym) const -> result_type { std::stringstream ss; ss << "{\"type\":\"" << mapnik::symbolizer_traits::name() << "\","; ss << "\"properties\":{"; bool first = true; for (auto const& prop : sym.properties) { auto const& meta = mapnik::get_meta(prop.first); if (first) first = false; else ss << ","; ss << "\"" << std::get<0>(meta) << "\":"; ss << boost::apply_visitor(mapnik::symbolizer_property_value_string(meta),prop.second); } ss << "}}"; return ss.str(); } }; std::string __str__(mapnik::symbolizer const& sym) { return boost::apply_visitor(symbolizer_to_json(), sym); } std::string get_symbolizer_type(symbolizer const& sym) { return mapnik::symbolizer_name(sym); // FIXME - do we need this ? } const point_symbolizer& point_(symbolizer const& sym ) { return boost::get(sym); } const line_symbolizer& line_( const symbolizer& sym ) { return boost::get(sym); } const polygon_symbolizer& polygon_( const symbolizer& sym ) { return boost::get(sym); } const raster_symbolizer& raster_( const symbolizer& sym ) { return boost::get(sym); } const text_symbolizer& text_( const symbolizer& sym ) { return boost::get(sym); } const shield_symbolizer& shield_( const symbolizer& sym ) { return boost::get(sym); } const line_pattern_symbolizer& line_pattern_( const symbolizer& sym ) { return boost::get(sym); } const polygon_pattern_symbolizer& polygon_pattern_( const symbolizer& sym ) { return boost::get(sym); } const building_symbolizer& building_( const symbolizer& sym ) { return boost::get(sym); } const markers_symbolizer& markers_( const symbolizer& sym ) { return boost::get(sym); } struct symbolizer_hash_visitor : public boost::static_visitor { template std::size_t operator() (T const& sym) const { return mapnik::symbolizer_hash::value(sym); } }; std::size_t hash_impl(symbolizer const& sym) { return boost::apply_visitor(symbolizer_hash_visitor(), sym); } } void export_symbolizer() { using namespace boost::python; implicitly_convertible(); implicitly_convertible(); implicitly_convertible(); implicitly_convertible(); implicitly_convertible(); implicitly_convertible(); implicitly_convertible(); enum_("keys") .value("gamma", mapnik::keys::gamma) .value("gamma_method",mapnik::keys::gamma_method) ; class_("Symbolizer",no_init) .def("type",get_symbolizer_type) .def("__hash__",hash_impl) .def("point",point_, return_value_policy()) .def("line",line_, return_value_policy()) .def("line_pattern",line_pattern_, return_value_policy()) .def("polygon",polygon_, return_value_policy()) .def("polygon_pattern",polygon_pattern_, return_value_policy()) .def("raster",raster_, return_value_policy()) .def("shield",shield_, return_value_policy()) .def("text",text_, return_value_policy()) .def("building",building_, return_value_policy()) .def("markers",markers_, return_value_policy()) ; class_("SymbolizerBase",no_init) .def("__setitem__",&__setitem__) .def("__setattr__",&__setitem__) .def("__getitem__",&__getitem__) .def("__getattr__",&__getitem__) .def("__str__", &__str__) ; } void export_shield_symbolizer() { using namespace boost::python; class_< shield_symbolizer, bases >("ShieldSymbolizer", init<>("Default ctor")) ; } void export_polygon_symbolizer() { using namespace boost::python; class_ >("PolygonSymbolizer", init<>("Default ctor")) ; } void export_polygon_pattern_symbolizer() { using namespace boost::python; mapnik::enumeration_("pattern_alignment") .value("LOCAL",mapnik::LOCAL_ALIGNMENT) .value("GLOBAL",mapnik::GLOBAL_ALIGNMENT) ; class_("PolygonPatternSymbolizer", init<>("Default ctor")) ; } void export_raster_symbolizer() { using namespace boost::python; class_ >("RasterSymbolizer", init<>("Default ctor")) ; } void export_point_symbolizer() { using namespace boost::python; mapnik::enumeration_("point_placement") .value("CENTROID",mapnik::CENTROID_POINT_PLACEMENT) .value("INTERIOR",mapnik::INTERIOR_POINT_PLACEMENT) ; class_ >("PointSymbolizer", init<>("Default Point Symbolizer - 4x4 black square")) ; } void export_markers_symbolizer() { using namespace boost::python; mapnik::enumeration_("marker_placement") .value("POINT_PLACEMENT",mapnik::MARKER_POINT_PLACEMENT) .value("INTERIOR_PLACEMENT",mapnik::MARKER_INTERIOR_PLACEMENT) .value("LINE_PLACEMENT",mapnik::MARKER_LINE_PLACEMENT) ; mapnik::enumeration_("marker_multi_policy") .value("EACH",mapnik::MARKER_EACH_MULTI) .value("WHOLE",mapnik::MARKER_WHOLE_MULTI) .value("LARGEST",mapnik::MARKER_LARGEST_MULTI) ; class_ >("MarkersSymbolizer", init<>("Default Markers Symbolizer - circle")) ; } void export_line_symbolizer() { using namespace boost::python; mapnik::enumeration_("line_rasterizer") .value("FULL",mapnik::RASTERIZER_FULL) .value("FAST",mapnik::RASTERIZER_FAST) ; class_ >("LineSymbolizer", init<>("Default LineSymbolizer - 1px solid black")) ; } void export_line_pattern_symbolizer() { using namespace boost::python; class_ >("LinePatternSymbolizer", init<> ("Default LinePatternSymbolizer")) ; } void export_debug_symbolizer() { using namespace boost::python; mapnik::enumeration_("debug_symbolizer_mode") .value("COLLISION",mapnik::DEBUG_SYM_MODE_COLLISION) .value("VERTEX",mapnik::DEBUG_SYM_MODE_VERTEX) ; class_ >("DebugSymbolizer", init<>("Default debug Symbolizer")) ; } void export_building_symbolizer() { using namespace boost::python; class_ >("BuildingSymbolizer", init<>("Default BuildingSymbolizer")) ; }