mapnik/include/mapnik/symbolizer_utils.hpp
2022-11-10 15:57:38 +00:00

508 lines
15 KiB
C++

/*****************************************************************************
*
* This file is part of Mapnik (c++ mapping toolkit)
*
* Copyright (C) 2021 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
*
*****************************************************************************/
#ifndef MAPNIK_SYMBOLIZER_UTILS_HPP
#define MAPNIK_SYMBOLIZER_UTILS_HPP
// mapnik
#include <mapnik/symbolizer_keys.hpp>
#include <mapnik/raster_colorizer.hpp>
#include <mapnik/path_expression.hpp>
#include <mapnik/parse_path.hpp>
#include <mapnik/color.hpp>
#include <mapnik/expression.hpp>
#include <mapnik/color_factory.hpp>
#include <mapnik/symbolizer.hpp>
#include <mapnik/xml_tree.hpp>
#include <mapnik/config_error.hpp>
#include <mapnik/evaluate_global_attributes.hpp>
#include <mapnik/transform/transform_processor.hpp>
#include <mapnik/transform/parse_transform.hpp>
#include <mapnik/util/dasharray_parser.hpp>
#include <mapnik/util/variant.hpp>
namespace mapnik {
template<typename Symbolizer>
struct symbolizer_traits
{
static char const* name() { return "Unknown"; }
};
template<>
struct symbolizer_traits<point_symbolizer>
{
static char const* name() { return "PointSymbolizer"; }
};
template<>
struct symbolizer_traits<line_symbolizer>
{
static char const* name() { return "LineSymbolizer"; }
};
template<>
struct symbolizer_traits<polygon_symbolizer>
{
static char const* name() { return "PolygonSymbolizer"; }
};
template<>
struct symbolizer_traits<text_symbolizer>
{
static char const* name() { return "TextSymbolizer"; }
};
template<>
struct symbolizer_traits<line_pattern_symbolizer>
{
static char const* name() { return "LinePatternSymbolizer"; }
};
template<>
struct symbolizer_traits<polygon_pattern_symbolizer>
{
static char const* name() { return "PolygonPatternSymbolizer"; }
};
template<>
struct symbolizer_traits<markers_symbolizer>
{
static char const* name() { return "MarkersSymbolizer"; }
};
template<>
struct symbolizer_traits<shield_symbolizer>
{
static char const* name() { return "ShieldSymbolizer"; }
};
template<>
struct symbolizer_traits<raster_symbolizer>
{
static char const* name() { return "RasterSymbolizer"; }
};
template<>
struct symbolizer_traits<building_symbolizer>
{
static char const* name() { return "BuildingSymbolizer"; }
};
template<>
struct symbolizer_traits<group_symbolizer>
{
static char const* name() { return "GroupSymbolizer"; }
};
template<>
struct symbolizer_traits<debug_symbolizer>
{
static char const* name() { return "DebugSymbolizer"; }
};
template<>
struct symbolizer_traits<dot_symbolizer>
{
static char const* name() { return "DotSymbolizer"; }
};
// symbolizer name impl
namespace detail {
struct symbolizer_name_impl
{
public:
template<typename Symbolizer>
std::string operator()(Symbolizer const&) const
{
return symbolizer_traits<Symbolizer>::name();
}
};
} // namespace detail
inline std::string symbolizer_name(symbolizer const& sym)
{
std::string type = util::apply_visitor(detail::symbolizer_name_impl(), sym);
return type;
}
// https://github.com/mapnik/mapnik/issues/2324
/*
template <typename Meta>
class symbolizer_property_value_string
{
public:
symbolizer_property_value_string (Meta const& meta)
: meta_(meta) {}
std::string operator() ( mapnik::enumeration_wrapper const& e) const
{
std::stringstream ss;
auto const& convert_fun_ptr(std::get<1>(meta_));
if ( convert_fun_ptr )
{
ss << convert_fun_ptr(e);
}
return ss.str();
}
std::string operator () ( path_expression_ptr const& expr) const
{
std::ostringstream ss;
if (expr)
{
ss << '\"' << path_processor::to_string(*expr) << '\"';
}
return ss.str();
}
std::string operator () (text_placements_ptr const& expr) const
{
return std::string("\"<fixme-text-placement-ptr>\"");
}
std::string operator () (raster_colorizer_ptr const& expr) const
{
return std::string("\"<fixme-raster-colorizer-ptr>\"");
}
std::string operator () (transform_type const& expr) const
{
std::ostringstream ss;
if (expr)
{
ss << '\"' << transform_processor_type::to_string(*expr) << '\"';
}
return ss.str();
}
std::string operator () (expression_ptr const& expr) const
{
std::ostringstream ss;
if (expr)
{
ss << '\"' << "FIXME" << '\"';
}
return ss.str();
}
std::string operator () (color const& c) const
{
std::ostringstream ss;
ss << '\"' << c << '\"';
return ss.str();
}
std::string operator () (dash_array const& dash) const
{
std::ostringstream ss;
for (std::size_t i = 0; i < dash.size(); ++i)
{
ss << dash[i].first << ", " << dash[i].second;
if ( i + 1 < dash.size() ) ss << ',';
}
return ss.str();
}
template <typename T>
std::string operator () ( T const& val ) const
{
std::ostringstream ss;
ss << val;
return ss.str();
}
private:
Meta const& meta_;
};
struct symbolizer_to_json
{
using result_type = std::string;
template <typename T>
auto operator() (T const& sym) const -> result_type
{
std::stringstream ss;
ss << "{\"type\":\"" << mapnik::symbolizer_traits<T>::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 << util::apply_visitor(symbolizer_property_value_string<property_meta_type>(meta),prop.second);
}
ss << "}}";
return ss.str();
}
};
*/
namespace {
template<typename Symbolizer, typename T>
struct set_property_impl
{
static void apply(Symbolizer&, mapnik::keys, std::string const&) { std::cerr << "do nothing" << std::endl; }
};
template<typename Symbolizer>
struct set_property_impl<Symbolizer, std::integral_constant<property_types, property_types::target_color>>
{
static void apply(Symbolizer& sym, mapnik::keys key, std::string const& val)
{
put(sym, key, mapnik::parse_color(val));
}
};
template<typename Symbolizer>
struct set_property_impl<Symbolizer, std::integral_constant<property_types, property_types::target_double>>
{
static void apply(Symbolizer&, mapnik::keys, std::string const&) { std::cerr << " expects double" << std::endl; }
};
template<typename Symbolizer>
struct set_property_impl<Symbolizer, std::integral_constant<property_types, property_types::target_bool>>
{
static void apply(Symbolizer&, mapnik::keys, std::string const&) { std::cerr << " expects bool" << std::endl; }
};
} // namespace
template<typename Symbolizer, typename T>
inline void set_property(Symbolizer& sym, mapnik::keys key, T const& val)
{
switch (std::get<2>(get_meta(key)))
{
case property_types::target_bool:
set_property_impl<Symbolizer, std::integral_constant<property_types, property_types::target_bool>>::apply(
sym,
key,
val);
break;
case property_types::target_integer:
set_property_impl<Symbolizer,
std::integral_constant<property_types, property_types::target_integer>>::apply(sym,
key,
val);
break;
case property_types::target_double:
set_property_impl<Symbolizer, std::integral_constant<property_types, property_types::target_double>>::apply(
sym,
key,
val);
break;
case property_types::target_color:
set_property_impl<Symbolizer, std::integral_constant<property_types, property_types::target_color>>::apply(
sym,
key,
val);
break;
default:
break;
}
}
template<typename Symbolizer, typename T>
inline void set_property_from_value(Symbolizer& sym, mapnik::keys key, T const& val)
{
switch (std::get<2>(get_meta(key)))
{
case property_types::target_bool:
put(sym, key, val.to_bool());
break;
case property_types::target_integer:
put(sym, key, val.to_int());
break;
case property_types::target_double:
put(sym, key, val.to_double());
break;
case property_types::target_color:
put(sym, key, mapnik::parse_color(val.to_string()));
break;
default:
break;
}
}
namespace detail {
// helpers
template<typename Symbolizer, typename T, bool is_enum = false>
struct set_symbolizer_property_impl
{
static void apply(Symbolizer& sym, keys key, std::string const& name, xml_node const& node)
{
using value_type = T;
try
{
boost::optional<value_type> val = node.get_opt_attr<value_type>(name);
if (val)
put(sym, key, *val);
}
catch (config_error const& ex)
{
// try parsing as an expression
boost::optional<expression_ptr> val = node.get_opt_attr<expression_ptr>(name);
if (val)
{
// first try pre-evaluate expressions which don't have dynamic properties
auto result = pre_evaluate_expression<mapnik::value>(*val);
if (std::get<1>(result))
{
set_property_from_value(sym, key, std::get<0>(result));
}
else
{
// expression_ptr
put(sym, key, *val);
}
}
else
{
throw;
}
}
}
};
template<typename Symbolizer>
struct set_symbolizer_property_impl<Symbolizer, transform_type, false>
{
static void apply(Symbolizer& sym, keys key, std::string const& name, xml_node const& node)
{
boost::optional<std::string> transform = node.get_opt_attr<std::string>(name);
if (transform)
put(sym, key, mapnik::parse_transform(*transform));
}
};
template<typename Symbolizer>
struct set_symbolizer_property_impl<Symbolizer, dash_array, false>
{
static void apply(Symbolizer& sym, keys key, std::string const& name, xml_node const& node)
{
boost::optional<std::string> str = node.get_opt_attr<std::string>(name);
if (str)
{
dash_array dash;
if (util::parse_dasharray(*str, dash))
{
put(sym, key, dash);
}
else
{
boost::optional<expression_ptr> val = node.get_opt_attr<expression_ptr>(name);
if (val)
{
// first try pre-evaluate expressions which don't have dynamic properties
auto result = pre_evaluate_expression<mapnik::value>(*val);
if (std::get<1>(result))
{
set_property_from_value(sym, key, std::get<0>(result));
}
else
{
// expression_ptr
put(sym, key, *val);
}
}
else
{
throw config_error(std::string("Failed to parse dasharray ") + "'. Expected a " +
"list of floats or 'none' but got '" + (*str) + "'");
}
}
}
}
};
template<typename Symbolizer, typename T>
struct set_symbolizer_property_impl<Symbolizer, T, true>
{
static void apply(Symbolizer& sym, keys key, std::string const& name, xml_node const& node)
{
using value_type = T;
boost::optional<std::string> enum_str = node.get_opt_attr<std::string>(name);
if (enum_str)
{
boost::optional<value_type> enum_val = detail::enum_traits<value_type>::from_string(*enum_str);
if (enum_val)
{
put(sym, key, *enum_val);
}
else
{
boost::optional<expression_ptr> val = node.get_opt_attr<expression_ptr>(name);
if (val)
{
// first try pre-evaluating expression
auto result = pre_evaluate_expression<value>(*val);
if (std::get<1>(result))
{
boost::optional<value_type> enum_val2 =
detail::enum_traits<value_type>::from_string(std::get<0>(result).to_string());
if (enum_val2)
{
put(sym, key, *enum_val);
}
else
{
// can't evaluate
throw config_error("failed to parse '" + name + "'");
}
}
else
{
// put expression_ptr
put(sym, key, *val);
}
}
else
{
throw config_error("failed to parse '" + name + "'");
}
}
}
}
};
} // namespace detail
template<typename Symbolizer, typename T>
void set_symbolizer_property(Symbolizer& sym, keys key, xml_node const& node)
{
std::string const& name = std::get<0>(get_meta(key));
if (node.has_attribute(name))
{
detail::set_symbolizer_property_impl<Symbolizer, T, std::is_enum<T>::value>::apply(sym, key, name, node);
}
}
} // namespace mapnik
#endif // MAPNIK_SYMBOLIZER_UTILS_HPP