/***************************************************************************** * * 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 #include "compare_images.hpp" // stl #include #include #include // mapnik #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"; unsigned compare(image_type const & actual, boost::filesystem::path const& reference) const { return compare_images(actual, reference.string()); } 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"; 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::fabs(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 class renderer { public: 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 { typename Renderer::image_type image(ren.render(map, scale_factor)); boost::filesystem::path reference = reference_dir / image_file_name(name, map.width(), map.height(), scale_factor, true, Renderer::ext); 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 = map_size(map.width(), map.height()); 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, map.width(), map.height(), scale_factor, false, Renderer::ext); 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, double width, double height, double scale_factor, bool reference, std::string const& ext) const { std::stringstream s; s << test_name << '-' << width << '-' << height << '-' << std::fixed << std::setprecision(1) << scale_factor << '-' << Renderer::name << (reference ? "-reference" : "") << ext; return s.str(); } const Renderer ren; boost::filesystem::path const & output_dir; boost::filesystem::path const & reference_dir; const bool overwrite; }; } #endif