-
Notifications
You must be signed in to change notification settings - Fork 319
/
Copy pathNavigationMapView+IntersectionAnnotations.swift
133 lines (109 loc) · 4.22 KB
/
NavigationMapView+IntersectionAnnotations.swift
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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
import CoreLocation
import UIKit
import MapboxDirections
import MapboxCoreNavigation
import Turf
import MapboxMaps
extension NavigationMapView {
struct EdgeIntersection {
var root: RoadGraph.Edge
var branch: RoadGraph.Edge
var rootMetadata: RoadGraph.Edge.Metadata
var rootShape: LineString
var branchMetadata: RoadGraph.Edge.Metadata
var branchShape: LineString
var coordinate: CLLocationCoordinate2D? {
rootShape.coordinates.first
}
var annotationPoint: CLLocationCoordinate2D? {
guard let length = branchShape.distance() else { return nil }
let targetDistance = min(length / 2, Double.random(in: 15...30))
guard let annotationPoint = branchShape.coordinateFromStart(distance: targetDistance) else { return nil }
return annotationPoint
}
var wayName: String? {
guard let roadName = rootMetadata.names.first else { return nil }
switch roadName {
case .name(let name):
return name
case .code(let code):
return "(\(code))"
}
}
var intersectingWayName: String? {
guard let roadName = branchMetadata.names.first else { return nil }
switch roadName {
case .name(let name):
return name
case .code(let code):
return "(\(code))"
}
}
var incidentAngle: CLLocationDegrees {
return (branchMetadata.heading - rootMetadata.heading).wrap(min: 0, max: 360)
}
var description: String {
return "EdgeIntersection: root: \(wayName ?? "") intersection: \(intersectingWayName ?? "") coordinate: \(String(describing: coordinate))"
}
}
enum AnnotationTailPosition: Int {
case left
case right
case center
}
class AnnotationCacheEntry: Equatable, Hashable {
var wayname: String
var coordinate: CLLocationCoordinate2D
var intersection: EdgeIntersection?
var feature: Feature
var lastAccessTime: Date
init(coordinate: CLLocationCoordinate2D, wayname: String, intersection: EdgeIntersection? = nil, feature: Feature) {
self.wayname = wayname
self.coordinate = coordinate
self.intersection = intersection
self.feature = feature
self.lastAccessTime = Date()
}
static func == (lhs: AnnotationCacheEntry, rhs: AnnotationCacheEntry) -> Bool {
return lhs.wayname == rhs.wayname
}
func hash(into hasher: inout Hasher) {
hasher.combine(wayname.hashValue)
}
}
class AnnotationCache {
private let maxEntryAge = TimeInterval(30)
var entries = Set<AnnotationCacheEntry>()
var cachePruningTimer: Timer?
init() {
// periodically prune the cache to remove entries that have been passed already
cachePruningTimer = Timer.scheduledTimer(withTimeInterval: 15, repeats: true, block: { [weak self] _ in
self?.prune()
})
}
deinit {
cachePruningTimer?.invalidate()
cachePruningTimer = nil
}
func setValue(feature: Feature, coordinate: CLLocationCoordinate2D, intersection: EdgeIntersection?, for wayname: String) {
entries.insert(AnnotationCacheEntry(coordinate: coordinate, wayname: wayname, intersection: intersection, feature: feature))
}
func value(for wayname: String) -> AnnotationCacheEntry? {
let matchingEntry = entries.first { entry -> Bool in
entry.wayname == wayname
}
if let matchingEntry = matchingEntry {
// update the timestamp used for pruning the cache
matchingEntry.lastAccessTime = Date()
}
return matchingEntry
}
private func prune() {
let now = Date()
entries.filter { now.timeIntervalSince($0.lastAccessTime) > maxEntryAge }.forEach { remove($0) }
}
public func remove(_ entry: AnnotationCacheEntry) {
entries.remove(entry)
}
}
}