/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2015 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 #if defined(GRID_RENDERER) #include #endif #if defined(HAVE_CAIRO) #include #include #endif #if defined(SVG_RENDERER) #include #endif // boost #include namespace visual_tests { template struct 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 agg_renderer : 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 : 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; } }; #endif #if defined(SVG_RENDERER) struct svg_renderer : renderer_base { static constexpr const char * name = "svg"; static constexpr const char * ext = ".svg"; static constexpr const bool support_tiles = false; 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(); } 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.is_open()) { throw std::runtime_error("could not open: '" + reference.string() + "'"); } std::string expected(std::istreambuf_iterator(stream.rdbuf()),(std::istreambuf_iterator())); stream.close(); 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((std::string("cannot open file for writing file ") + path.string()).c_str()); } else { file << image; file.close(); } } }; #endif #if defined(GRID_RENDERER) struct grid_renderer : 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) { } result test(std::string const & name, mapnik::Map const & map, double scale_factor) const { image_type image(ren.render(map, scale_factor)); return report(image, name, { map.width(), map.height() }, { 1, 1 }, scale_factor); } result test_tiles(std::string const & name, mapnik::Map & map, map_size const & tiles, double scale_factor) const { image_type image(map.width(), map.height()); mapnik::box2d box = map.get_current_extent(); 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 report(image, name, { image.width(), image.height() }, tiles, scale_factor); } 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; boost::filesystem::path const & output_dir; boost::filesystem::path const & reference_dir; const bool overwrite; }; } #endif