summaryrefslogtreecommitdiff
path: root/test/map/prefetch.test.cpp
blob: c3ff04acfcf65de8569051c64ce3a2ab82631a53 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
#include <mbgl/test/util.hpp>
#include <mbgl/test/stub_file_source.hpp>

#include <mbgl/gl/headless_backend.hpp>
#include <mbgl/gl/offscreen_view.hpp>
#include <mbgl/map/map.hpp>
#include <mbgl/renderer/renderer.hpp>
#include <mbgl/renderer/backend_scope.hpp>
#include <mbgl/storage/default_file_source.hpp>
#include <mbgl/style/style.hpp>
#include <mbgl/util/default_thread_pool.hpp>
#include <mbgl/util/image.hpp>
#include <mbgl/util/io.hpp>
#include <mbgl/util/run_loop.hpp>
#include <mbgl/test/stub_renderer_frontend.hpp>

#include <algorithm>
#include <string>
#include <vector>

using namespace mbgl;
using namespace mbgl::style;
using namespace std::literals::string_literals;

TEST(Map, PrefetchTiles) {
    util::RunLoop runLoop;
    HeadlessBackend backend(test::sharedDisplay());
    BackendScope scope(backend);
    OffscreenView view(backend.getContext(), { 512, 512 });
    ThreadPool threadPool(4);
    StubFileSource fileSource;
    StubRendererFrontend rendererFrontend {
            std::make_unique<Renderer>(backend, 1, fileSource, threadPool), view };
    Map map(rendererFrontend, MapObserver::nullObserver(), view.getSize(), 1, fileSource, threadPool, MapMode::Still);

    std::vector<int> tiles;

    fileSource.response = [&] (const Resource& res) -> optional<Response> {
        Response response;

        auto zoom = std::stoi(res.url);
        tiles.push_back(zoom);

        // Return a red tile for prefetched tiles or green to the actual tile.
        // The end rendering result should be all green because the map is only
        // considered fully rendered when only ideal tiles are shown.
        if (zoom == int(map.getZoom()) + 1) {
            response.data = std::make_shared<std::string>(
                util::read_file("test/fixtures/map/prefetch/tile_green.png"));
        } else {
            response.data = std::make_shared<std::string>(
                util::read_file("test/fixtures/map/prefetch/tile_red.png"));
        }

        return { std::move(response) };
    };

    auto checkTilesForZoom = [&](int zoom, const std::vector<int>& expected) {
        tiles.clear();

        // Force tile reloading.
        map.getStyle().loadJSON(util::read_file("test/fixtures/map/prefetch/empty.json"));
        map.getStyle().loadJSON(util::read_file("test/fixtures/map/prefetch/style.json"));

        map.setLatLngZoom({ 40.726989, -73.992857 }, zoom); // Manhattan

        // Should always render the ideal tiles (i.e. a green map)
        test::checkImage("test/fixtures/map/prefetch", test::render(map, view));

        ASSERT_TRUE(std::is_permutation(tiles.begin(), tiles.end(), expected.begin()));
        ASSERT_FALSE(tiles.empty());
    };

    // Check defaults, should be 4.
    ASSERT_EQ(map.getPrefetchZoomDelta(), 4);
    checkTilesForZoom(12, { 13, 13, 13, 13, 13, 13, 13, 13, 13, 9 });

    // Setting it to 0 disables prefetching.
    map.setPrefetchZoomDelta(0);

    // No prefetching, raster tiles will use ideal
    // tiles instead of the actual zoom level, that is
    // why the zoom levels for non-prefetched tiles are
    // not the same.
    checkTilesForZoom(10, { 11, 11, 11, 11, 11, 11, 11, 11, 11 });

    map.setPrefetchZoomDelta(5);
    checkTilesForZoom(12, { 13, 13, 13, 13, 13, 13, 13, 13, 13, 8 });

    // Should clamp at `minzoom`.
    map.setPrefetchZoomDelta(20);
    checkTilesForZoom(10, { 11, 11, 11, 11, 11, 11, 11, 11, 11, 0 });

    // Disabled again.
    map.setPrefetchZoomDelta(0);
    checkTilesForZoom(13, { 14, 14, 14, 14, 14, 14, 14, 14, 14 });
}