/***************************************************************************** * * 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 RENDERER_HPP #define RENDERER_HPP // stl #include #include #include #include // mapnik #include #include #include #include #include #if defined(GRID_RENDERER) #include #endif #if defined(HAVE_CAIRO) #include #include #ifdef CAIRO_HAS_SVG_SURFACE #include #endif #ifdef CAIRO_HAS_PS_SURFACE #include #endif #ifdef CAIRO_HAS_PDF_SURFACE #include #endif #endif #if defined(SVG_RENDERER) #include #endif // boost #include namespace visual_tests { template struct raster_renderer_base { using image_type = ImageType; static constexpr const char * ext = ".png"; static constexpr const bool support_tiles = true; unsigned compare(image_type const & actual, boost::filesystem::path const& reference) const { std::unique_ptr reader(mapnik::get_image_reader(reference.string(), "png")); if (!reader.get()) { throw mapnik::image_reader_exception("Failed to load: " + reference.string()); } mapnik::image_any ref_image_any = reader->read(0, 0, reader->width(), reader->height()); ImageType const & reference_image = mapnik::util::get(ref_image_any); return mapnik::compare(actual, reference_image, 0, true); } void save(image_type const & image, boost::filesystem::path const& path) const { mapnik::save_to_file(image, path.string(), "png32"); } }; struct vector_renderer_base { using image_type = std::string; static constexpr const bool support_tiles = false; unsigned compare(image_type const & actual, boost::filesystem::path const& reference) const { std::ifstream stream(reference.string().c_str(), std::ios_base::in | std::ios_base::binary); if (!stream) { throw std::runtime_error("Could not open: " + reference.string()); } std::string expected(std::istreambuf_iterator(stream.rdbuf()), std::istreambuf_iterator()); return std::max(actual.size(), expected.size()) - std::min(actual.size(), expected.size()); } void save(image_type const & image, boost::filesystem::path const& path) const { std::ofstream file(path.string().c_str(), std::ios::out | std::ios::trunc | std::ios::binary); if (!file) { throw std::runtime_error("Cannot open file for writing: " + path.string()); } file << image; } }; struct agg_renderer : raster_renderer_base { static constexpr const char * name = "agg"; image_type render(mapnik::Map const & map, double scale_factor) const { image_type image(map.width(), map.height()); mapnik::agg_renderer ren(map, image, scale_factor); ren.apply(); return image; } }; #if defined(HAVE_CAIRO) struct cairo_renderer : raster_renderer_base { static constexpr const char * name = "cairo"; image_type render(mapnik::Map const & map, double scale_factor) const { mapnik::cairo_surface_ptr image_surface( cairo_image_surface_create(CAIRO_FORMAT_ARGB32, map.width(), map.height()), mapnik::cairo_surface_closer()); mapnik::cairo_ptr image_context(mapnik::create_context(image_surface)); mapnik::cairo_renderer ren(map, image_context, scale_factor); ren.apply(); image_type image(map.width(), map.height()); mapnik::cairo_image_to_rgba8(image, image_surface); return image; } }; using surface_create_type = cairo_surface_t *(&)(cairo_write_func_t, void *, double, double); template struct cairo_vector_renderer : vector_renderer_base { static cairo_status_t write(void *closure, const unsigned char *data, unsigned int length) { std::ostringstream & ss = *reinterpret_cast(closure); ss.write(reinterpret_cast(data), length); return ss ? CAIRO_STATUS_SUCCESS : CAIRO_STATUS_WRITE_ERROR; } image_type render(mapnik::Map const & map, double scale_factor) const { std::ostringstream ss(std::stringstream::binary); mapnik::cairo_surface_ptr image_surface( SurfaceCreateFunction(write, &ss, map.width(), map.height()), mapnik::cairo_surface_closer()); mapnik::cairo_ptr image_context(mapnik::create_context(image_surface)); mapnik::cairo_renderer ren(map, image_context, scale_factor); ren.apply(); cairo_surface_finish(&*image_surface); return ss.str(); } }; #ifdef CAIRO_HAS_SVG_SURFACE inline cairo_surface_t *create_svg_1_2(cairo_write_func_t write_func, void *closure, double width, double height) { cairo_surface_t * surface = cairo_svg_surface_create_for_stream(write_func, closure, width, height); cairo_svg_surface_restrict_to_version(surface, CAIRO_SVG_VERSION_1_2); return surface; } struct cairo_svg_renderer : cairo_vector_renderer { static constexpr const char * name = "cairo-svg"; static constexpr const char * ext = ".svg"; }; #endif #ifdef CAIRO_HAS_PS_SURFACE struct cairo_ps_renderer : cairo_vector_renderer { static constexpr const char * name = "cairo-ps"; static constexpr const char * ext = ".ps"; }; #endif #ifdef CAIRO_HAS_PDF_SURFACE struct cairo_pdf_renderer : cairo_vector_renderer { static constexpr const char * name = "cairo-pdf"; static constexpr const char * ext = ".pdf"; }; #endif #endif #if defined(SVG_RENDERER) struct svg_renderer : vector_renderer_base { static constexpr const char * name = "svg"; static constexpr const char * ext = ".svg"; image_type render(mapnik::Map const & map, double scale_factor) const { std::stringstream ss; std::ostream_iterator output_stream_iterator(ss); mapnik::svg_renderer> ren(map, output_stream_iterator, scale_factor); ren.apply(); return ss.str(); } }; #endif #if defined(GRID_RENDERER) struct grid_renderer : raster_renderer_base { static constexpr const char * name = "grid"; void convert(mapnik::grid::data_type const & grid, image_type & image) const { for (std::size_t y = 0; y < grid.height(); ++y) { mapnik::grid::value_type const * grid_row = grid.get_row(y); image_type::pixel_type * image_row = image.get_row(y); for (std::size_t x = 0; x < grid.width(); ++x) { mapnik::grid::value_type val = grid_row[x]; if (val == mapnik::grid::base_mask) { image_row[x] = 0; continue; } if (val < 0) { throw std::runtime_error("grid renderer: feature id is negative."); } val *= 100; if (val > 0x00ffffff) { throw std::runtime_error("grid renderer: feature id is too high."); } image_row[x] = val | 0xff000000; } } } image_type render(mapnik::Map const & map, double scale_factor) const { mapnik::grid grid(map.width(), map.height(), "__id__"); mapnik::grid_renderer ren(map, grid, scale_factor); ren.apply(); image_type image(map.width(), map.height()); convert(grid.data(), image); return image; } }; #endif template void set_rectangle(T const & src, T & dst, std::size_t x, std::size_t y) { mapnik::box2d ext0(0, 0, dst.width(), dst.height()); mapnik::box2d ext1(x, y, x + src.width(), y + src.height()); if (ext0.intersects(ext1)) { mapnik::box2d box = ext0.intersect(ext1); for (std::size_t pix_y = box.miny(); pix_y < static_cast(box.maxy()); ++pix_y) { typename T::pixel_type * row_to = dst.get_row(pix_y); typename T::pixel_type const * row_from = src.get_row(pix_y - y); for (std::size_t pix_x = box.minx(); pix_x < static_cast(box.maxx()); ++pix_x) { row_to[pix_x] = row_from[pix_x - x]; } } } } template class renderer { public: using renderer_type = Renderer; using image_type = typename Renderer::image_type; renderer(boost::filesystem::path const & _output_dir, boost::filesystem::path const & _reference_dir, bool _overwrite) : ren(), output_dir(_output_dir), reference_dir(_reference_dir), overwrite(_overwrite) { } image_type render(mapnik::Map const & map, double scale_factor) const { return ren.render(map, scale_factor); } image_type render(mapnik::Map & map, double scale_factor, map_size const & tiles) const { mapnik::box2d box = map.get_current_extent(); image_type image(map.width(), map.height()); map.resize(image.width() / tiles.width, image.height() / tiles.height); double tile_box_width = box.width() / tiles.width; double tile_box_height = box.height() / tiles.height; for (std::size_t tile_y = 0; tile_y < tiles.height; tile_y++) { for (std::size_t tile_x = 0; tile_x < tiles.width; tile_x++) { mapnik::box2d tile_box( box.minx() + tile_x * tile_box_width, box.miny() + tile_y * tile_box_height, box.minx() + (tile_x + 1) * tile_box_width, box.miny() + (tile_y + 1) * tile_box_height); map.zoom_to_box(tile_box); image_type tile(ren.render(map, scale_factor)); set_rectangle(tile, image, tile_x * tile.width(), (tiles.height - 1 - tile_y) * tile.height()); } } return image; } result report(image_type const & image, std::string const & name, map_size const & size, map_size const & tiles, double scale_factor) const { boost::filesystem::path reference = reference_dir / image_file_name(name, size, tiles, scale_factor, true); bool reference_exists = boost::filesystem::exists(reference); result res; res.state = reference_exists ? STATE_OK : STATE_OVERWRITE; res.name = name; res.renderer_name = Renderer::name; res.scale_factor = scale_factor; res.size = size; res.tiles = tiles; res.reference_image_path = reference; res.diff = reference_exists ? ren.compare(image, reference) : 0; if (res.diff) { boost::filesystem::create_directories(output_dir); boost::filesystem::path path = output_dir / image_file_name(name, size, tiles, scale_factor, false); res.actual_image_path = path; res.state = STATE_FAIL; ren.save(image, path); } if ((res.diff && overwrite) || !reference_exists) { ren.save(image, reference); res.state = STATE_OVERWRITE; } return res; } private: std::string image_file_name(std::string const & test_name, map_size const & size, map_size const & tiles, double scale_factor, bool reference) const { std::stringstream s; s << test_name << '-' << size.width << '-' << size.height << '-'; if (tiles.width > 1 || tiles.height > 1) { s << tiles.width << 'x' << tiles.height << '-'; } s << std::fixed << std::setprecision(1) << scale_factor << '-' << Renderer::name; if (reference) { s << "-reference"; } s << Renderer::ext; return s.str(); } const Renderer ren; const boost::filesystem::path output_dir; const boost::filesystem::path reference_dir; const bool overwrite; }; using renderer_type = mapnik::util::variant #if defined(HAVE_CAIRO) ,renderer #ifdef CAIRO_HAS_SVG_SURFACE ,renderer #endif #ifdef CAIRO_HAS_PS_SURFACE ,renderer #endif #ifdef CAIRO_HAS_PDF_SURFACE ,renderer #endif #endif #if defined(SVG_RENDERER) ,renderer #endif #if defined(GRID_RENDERER) ,renderer #endif >; } #endif