123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488 |
- // Boost.Geometry (aka GGL, Generic Geometry Library)
- // Copyright (c) 2009-2015 Barend Gehrels, Amsterdam, the Netherlands.
- // Copyright (c) 2023 Adam Wulkiewicz, Lodz, Poland.
- // This file was modified by Oracle on 2015-2021.
- // Modifications copyright (c) 2015-2020, Oracle and/or its affiliates.
- // Contributed and/or modified by Menelaos Karavelas, on behalf of Oracle
- // Contributed and/or modified by Adam Wulkiewicz, on behalf of Oracle
- // Parts of Boost.Geometry are redesigned from Geodan's Geographic Library
- // (geolib/GGL), copyright (c) 1995-2010 Geodan, Amsterdam, the Netherlands.
- // Use, modification and distribution is subject to the Boost Software License,
- // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
- // http://www.boost.org/LICENSE_1_0.txt)
- #ifndef BOOST_GEOMETRY_IO_SVG_MAPPER_HPP
- #define BOOST_GEOMETRY_IO_SVG_MAPPER_HPP
- #include <cstdio>
- #include <memory>
- #include <type_traits>
- #include <vector>
- #include <boost/algorithm/string/classification.hpp>
- #include <boost/algorithm/string/split.hpp>
- #include <boost/config.hpp>
- #include <boost/noncopyable.hpp>
- #include <boost/geometry/core/static_assert.hpp>
- #include <boost/geometry/core/tags.hpp>
- #include <boost/geometry/core/tag_cast.hpp>
- #include <boost/geometry/algorithms/envelope.hpp>
- #include <boost/geometry/algorithms/expand.hpp>
- #include <boost/geometry/algorithms/is_empty.hpp>
- #include <boost/geometry/algorithms/transform.hpp>
- #include <boost/geometry/strategies/transform/map_transformer.hpp>
- #include <boost/geometry/views/segment_view.hpp>
- #include <boost/geometry/io/svg/write.hpp>
- // Helper geometries (all points are transformed to svg-points)
- #include <boost/geometry/geometries/geometries.hpp>
- namespace boost { namespace geometry
- {
- #ifndef DOXYGEN_NO_DISPATCH
- namespace dispatch
- {
- template <typename GeometryTag, typename Geometry, typename SvgPoint>
- struct svg_map
- {
- BOOST_GEOMETRY_STATIC_ASSERT_FALSE(
- "Not or not yet implemented for this Geometry type.",
- GeometryTag, Geometry);
- };
- template <typename Point, typename SvgPoint>
- struct svg_map<point_tag, Point, SvgPoint>
- {
- template <typename TransformStrategy>
- static inline void apply(std::ostream& stream,
- std::string const& style, double size,
- Point const& point, TransformStrategy const& strategy)
- {
- SvgPoint ipoint;
- geometry::transform(point, ipoint, strategy);
- stream << geometry::svg(ipoint, style, size) << std::endl;
- }
- };
- template <typename BoxSeg1, typename BoxSeg2, typename SvgPoint>
- struct svg_map_box_seg
- {
- template <typename TransformStrategy>
- static inline void apply(std::ostream& stream,
- std::string const& style, double size,
- BoxSeg1 const& box_seg, TransformStrategy const& strategy)
- {
- BoxSeg2 ibox_seg;
- // Fix bug in gcc compiler warning for possible uninitialization
- #if defined(BOOST_GCC)
- geometry::assign_zero(ibox_seg);
- #endif
- geometry::transform(box_seg, ibox_seg, strategy);
- stream << geometry::svg(ibox_seg, style, size) << std::endl;
- }
- };
- template <typename Box, typename SvgPoint>
- struct svg_map<box_tag, Box, SvgPoint>
- : svg_map_box_seg<Box, model::box<SvgPoint>, SvgPoint>
- {};
- template <typename Segment, typename SvgPoint>
- struct svg_map<segment_tag, Segment, SvgPoint>
- : svg_map_box_seg<Segment, model::segment<SvgPoint>, SvgPoint>
- {};
- template <typename Range1, typename Range2, typename SvgPoint>
- struct svg_map_range
- {
- template <typename TransformStrategy>
- static inline void apply(std::ostream& stream,
- std::string const& style, double size,
- Range1 const& range, TransformStrategy const& strategy)
- {
- Range2 irange;
- geometry::transform(range, irange, strategy);
- stream << geometry::svg(irange, style, size) << std::endl;
- }
- };
- template <typename Ring, typename SvgPoint>
- struct svg_map<ring_tag, Ring, SvgPoint>
- : svg_map_range<Ring, model::ring<SvgPoint>, SvgPoint>
- {};
- template <typename Linestring, typename SvgPoint>
- struct svg_map<linestring_tag, Linestring, SvgPoint>
- : svg_map_range<Linestring, model::linestring<SvgPoint>, SvgPoint>
- {};
- template <typename Polygon, typename SvgPoint>
- struct svg_map<polygon_tag, Polygon, SvgPoint>
- {
- template <typename TransformStrategy>
- static inline void apply(std::ostream& stream,
- std::string const& style, double size,
- Polygon const& polygon, TransformStrategy const& strategy)
- {
- model::polygon<SvgPoint> ipoly;
- geometry::transform(polygon, ipoly, strategy);
- stream << geometry::svg(ipoly, style, size) << std::endl;
- }
- };
- template <typename Multi, typename SvgPoint>
- struct svg_map<multi_tag, Multi, SvgPoint>
- {
- typedef typename single_tag_of
- <
- typename geometry::tag<Multi>::type
- >::type stag;
- template <typename TransformStrategy>
- static inline void apply(std::ostream& stream,
- std::string const& style, double size,
- Multi const& multi, TransformStrategy const& strategy)
- {
- for (auto it = boost::begin(multi); it != boost::end(multi); ++it)
- {
- svg_map
- <
- stag,
- typename boost::range_value<Multi>::type,
- SvgPoint
- >::apply(stream, style, size, *it, strategy);
- }
- }
- };
- template <typename SvgPoint, typename Geometry>
- struct devarianted_svg_map
- {
- template <typename TransformStrategy>
- static inline void apply(std::ostream& stream,
- std::string const& style,
- double size,
- Geometry const& geometry,
- TransformStrategy const& strategy)
- {
- svg_map
- <
- typename tag_cast
- <
- typename tag<Geometry>::type,
- multi_tag
- >::type,
- typename std::remove_const<Geometry>::type,
- SvgPoint
- >::apply(stream, style, size, geometry, strategy);
- }
- };
- template <typename SvgPoint, BOOST_VARIANT_ENUM_PARAMS(typename T)>
- struct devarianted_svg_map<SvgPoint, variant<BOOST_VARIANT_ENUM_PARAMS(T)> >
- {
- template <typename TransformStrategy>
- struct visitor: static_visitor<void>
- {
- std::ostream& m_os;
- std::string const& m_style;
- double m_size;
- TransformStrategy const& m_strategy;
- visitor(std::ostream& os,
- std::string const& style,
- double size,
- TransformStrategy const& strategy)
- : m_os(os)
- , m_style(style)
- , m_size(size)
- , m_strategy(strategy)
- {}
- template <typename Geometry>
- inline void operator()(Geometry const& geometry) const
- {
- devarianted_svg_map<SvgPoint, Geometry>::apply(m_os, m_style, m_size, geometry, m_strategy);
- }
- };
- template <typename TransformStrategy>
- static inline void apply(std::ostream& stream,
- std::string const& style,
- double size,
- variant<BOOST_VARIANT_ENUM_PARAMS(T)> const& geometry,
- TransformStrategy const& strategy)
- {
- boost::apply_visitor(visitor<TransformStrategy>(stream, style, size, strategy), geometry);
- }
- };
- } // namespace dispatch
- #endif
- template <typename SvgPoint, typename Geometry, typename TransformStrategy>
- inline void svg_map(std::ostream& stream,
- std::string const& style, double size,
- Geometry const& geometry, TransformStrategy const& strategy)
- {
- dispatch::devarianted_svg_map<SvgPoint, Geometry>::apply(stream,
- style, size, geometry, strategy);
- }
- /*!
- \brief Helper class to create SVG maps
- \tparam Point Point type, for input geometries.
- \tparam SameScale Boolean flag indicating if horizontal and vertical scale should
- be the same. The default value is true
- \tparam SvgCoordinateType Coordinate type of SVG points. SVG is capable to
- use floating point coordinates. Therefore the default value is double
- \ingroup svg
- \qbk{[include reference/io/svg.qbk]}
- */
- template
- <
- typename Point,
- bool SameScale = true,
- typename SvgCoordinateType = double
- >
- class svg_mapper : boost::noncopyable
- {
- typedef model::point<SvgCoordinateType, 2, cs::cartesian> svg_point_type;
- typedef typename geometry::select_most_precise
- <
- typename coordinate_type<Point>::type,
- double
- >::type calculation_type;
- typedef strategy::transform::map_transformer
- <
- calculation_type,
- geometry::dimension<Point>::type::value,
- geometry::dimension<Point>::type::value,
- true,
- SameScale
- > transformer_type;
- model::box<Point> m_bounding_box;
- std::unique_ptr<transformer_type> m_matrix;
- std::ostream& m_stream;
- SvgCoordinateType const m_width;
- SvgCoordinateType const m_height;
- calculation_type const m_scale{1.0};
- void scale_bounding_box()
- {
- if (m_scale != 1.0 && m_scale > 0)
- {
- // Zoom out (scale < 1) or zoom in (scale > 1).
- // The default is 0.95, giving a small margin around geometries.
- auto& b = m_bounding_box;
- auto const w = geometry::get<1, 0>(b) - geometry::get<0, 0>(b);
- auto const h = geometry::get<1, 1>(b) - geometry::get<0, 1>(b);
- auto const& m = (std::max)(w, h) * (1.0 - m_scale);
- geometry::set<0, 0>(b, geometry::get<0, 0>(b) - m);
- geometry::set<0, 1>(b, geometry::get<0, 1>(b) - m);
- geometry::set<1, 0>(b, geometry::get<1, 0>(b) + m);
- geometry::set<1, 1>(b, geometry::get<1, 1>(b) + m);
- }
- }
- void init_matrix()
- {
- if (! m_matrix)
- {
- scale_bounding_box();
- m_matrix.reset(new transformer_type(m_bounding_box,
- m_width, m_height));
- }
- }
- void write_header(std::string const& width_height)
- {
- m_stream << "<?xml version=\"1.0\" standalone=\"no\"?>"
- << std::endl
- << "<!DOCTYPE svg PUBLIC \"-//W3C//DTD SVG 1.1//EN\""
- << std::endl
- << "\"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\">"
- << std::endl
- << "<svg " << width_height << " version=\"1.1\""
- << std::endl
- << "xmlns=\"http://www.w3.org/2000/svg\""
- << std::endl
- << "xmlns:xlink=\"http://www.w3.org/1999/xlink\""
- << ">"
- << std::endl;
- }
- public :
- /*!
- \brief Constructor, initializing the SVG map. Opens and initializes the SVG.
- Should be called explicitly.
- \param stream Output stream, should be a stream already open
- \param width Width of the SVG map (in SVG pixels)
- \param height Height of the SVG map (in SVG pixels)
- \param scale Scale factor of the automatically determined bounding box.
- If the factor is less than 1.0, there will be a margin around the
- geometries. A factor of 0.95 is often a convenient margin. If the
- factor is more than 1.0, the SVG map is zoomed and not all
- geometries will be visible.
- \param width_height Optional information to increase width and/or height
- */
- svg_mapper(std::ostream& stream
- , SvgCoordinateType width
- , SvgCoordinateType height
- , calculation_type scale
- , std::string const& width_height = "width=\"100%\" height=\"100%\"")
- : m_stream(stream)
- , m_width(width)
- , m_height(height)
- , m_scale(scale)
- {
- assign_inverse(m_bounding_box);
- write_header(width_height);
- }
- /*!
- \brief Constructor, initializing the SVG map. Opens and initializes the SVG.
- Should be called explicitly.
- \param stream Output stream, should be a stream already open
- \param width Width of the SVG map (in SVG pixels)
- \param height Height of the SVG map (in SVG pixels)
- \param width_height Optional information to increase width and/or height
- */
- svg_mapper(std::ostream& stream
- , SvgCoordinateType width
- , SvgCoordinateType height
- , std::string const& width_height = "width=\"100%\" height=\"100%\"")
- : m_stream(stream)
- , m_width(width)
- , m_height(height)
- {
- assign_inverse(m_bounding_box);
- write_header(width_height);
- }
- /*!
- \brief Destructor, called automatically. Closes the SVG by streaming <\/svg>
- */
- virtual ~svg_mapper()
- {
- m_stream << "</svg>" << std::endl;
- }
- /*!
- \brief Adds a geometry to the transformation matrix. After doing this,
- the specified geometry can be mapped fully into the SVG map
- \tparam Geometry \tparam_geometry
- \param geometry \param_geometry
- */
- template <typename Geometry>
- void add(Geometry const& geometry)
- {
- if (! geometry::is_empty(geometry))
- {
- expand(m_bounding_box,
- return_envelope
- <
- model::box<Point>
- >(geometry));
- }
- }
- /*!
- \brief Maps a geometry into the SVG map using the specified style
- \tparam Geometry \tparam_geometry
- \param geometry \param_geometry
- \param style String containing verbatim SVG style information
- \param size Optional size (used for SVG points) in SVG pixels. For linestrings,
- specify linewidth in the SVG style information
- */
- template <typename Geometry>
- void map(Geometry const& geometry, std::string const& style,
- double size = -1.0)
- {
- init_matrix();
- svg_map<svg_point_type>(m_stream, style, size, geometry, *m_matrix);
- }
- /*!
- \brief Adds a text to the SVG map
- \tparam TextPoint \tparam_point
- \param point Location of the text (in map units)
- \param s The text itself
- \param style String containing verbatim SVG style information, of the text
- \param offset_x Offset in SVG pixels, defaults to 0
- \param offset_y Offset in SVG pixels, defaults to 0
- \param lineheight Line height in SVG pixels, in case the text contains \n
- */
- template <typename TextPoint>
- void text(TextPoint const& point, std::string const& s,
- std::string const& style,
- double offset_x = 0.0, double offset_y = 0.0,
- double lineheight = 10.0)
- {
- init_matrix();
- svg_point_type map_point;
- transform(point, map_point, *m_matrix);
- m_stream
- << "<text style=\"" << style << "\""
- << " x=\"" << get<0>(map_point) + offset_x << "\""
- << " y=\"" << get<1>(map_point) + offset_y << "\""
- << ">";
- if (s.find('\n') == std::string::npos)
- {
- m_stream << s;
- }
- else
- {
- // Multi-line modus
- std::vector<std::string> split;
- boost::split(split, s, boost::is_any_of("\n"));
- for (auto const& item : split)
- {
- m_stream
- << "<tspan x=\"" << get<0>(map_point) + offset_x
- << "\""
- << " y=\"" << get<1>(map_point) + offset_y
- << "\""
- << ">" << item << "</tspan>";
- offset_y += lineheight;
- }
- }
- m_stream << "</text>" << std::endl;
- }
- };
- }} // namespace boost::geometry
- #endif // BOOST_GEOMETRY_IO_SVG_MAPPER_HPP
|