summaryrefslogtreecommitdiff
path: root/src/mbgl/util/tile_cover_impl.hpp
blob: 7c16718984ebf1ef38c19bfa483701638d0b6040 (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
#pragma once

#include <mbgl/util/tile_cover.hpp>
#include <mbgl/util/geometry.hpp>
#include <mbgl/util/optional.hpp>

#include <vector>
#include <map>
#include <queue>

namespace mbgl {

class TransformState;
class LatLngBounds;

namespace util {

struct Bound;

using Bounds = std::vector<Bound>;
using BoundsMap = std::map<uint32_t, Bounds>;

// A chain of points from a local minimum to a local maximum. `winding` indicates
//  the direction of the original geometry.
struct Bound {
    std::vector<Point<double>> points;
    size_t currentPoint = 0;
    bool winding = false;

    Bound() = default;
    Bound(const Bound& rhs) {
        points = rhs.points;
        currentPoint = rhs.currentPoint;
        winding = rhs.winding;
    }
    Bound& operator=(Bound&& rhs) {
        points = std::move(rhs.points);
        currentPoint = rhs.currentPoint;
        winding = rhs.winding;
        return *this;
    }

    // Compute the interpolated x coordinate at y for the current edge
    double interpolate(uint32_t y) {
        const auto& p0 = points[currentPoint];
        const auto& p1 = points[currentPoint + 1];

        const auto dx = p1.x - p0.x;
        const auto dy = p1.y - p0.y;
        auto x = p0.x;
        if (dx == 0) {
            return x;
        } else if (dy == 0){
            return y <= p0.y ? p0.x : p1.x;
        }
        if (y < p0.y) return x;
        if (y > p1.y) return p1.x;
        x = (dx / dy) * (y - p0.y) + p0.x;
        return x;
    }
};

class TileCover::Impl {
public:
    Impl(int32_t z, const Geometry<double>& geom, bool project = true);
    ~Impl() = default;

    optional<UnwrappedTileID> next();
    bool hasNext() const;

private:
    using TileSpans = std::queue<std::pair<int32_t, int32_t>>;

    void nextRow();

    const int32_t zoom;
    bool isClosed;

    BoundsMap boundsMap;
    BoundsMap::iterator currentBounds;
    // List of bounds that begin at or before `tileY`
    Bounds activeBounds;

    TileSpans tileXSpans;
    uint32_t tileY;
    int32_t tileX;
};

} // namespace util
} // namespace mbgl