-
Notifications
You must be signed in to change notification settings - Fork 55
/
bound.go
397 lines (324 loc) · 9.74 KB
/
bound.go
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
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
package geo
import (
"fmt"
"math"
"strings"
)
// A Bound represents an enclosed "box" in the 2D Euclidean or Cartesian plane.
// It does not know anything about the anti-meridian.
type Bound struct {
sw, ne *Point
}
// NewBound creates a new bound given the parameters.
func NewBound(west, east, south, north float64) *Bound {
return &Bound{
sw: &Point{math.Min(east, west), math.Min(north, south)},
ne: &Point{math.Max(east, west), math.Max(north, south)},
}
}
// NewBoundFromPoints creates a new bound given two opposite corners.
// These corners can be either sw/ne or se/nw.
func NewBoundFromPoints(corner, oppositeCorner *Point) *Bound {
b := &Bound{
sw: corner.Clone(),
ne: corner.Clone(),
}
b.Extend(oppositeCorner)
return b
}
// NewGeoBoundAroundPoint creates a new bound given a center point,
// and a distance from the center point in meters
func NewGeoBoundAroundPoint(center *Point, distance float64) *Bound {
if distance < 0 {
panic("invalid distance around center")
}
return geoBoundAroundPoint(center, distance)
}
// NewBoundFromMapTile creates a bound given an online map tile index.
// Panics if x or y is out of range for zoom level.
func NewBoundFromMapTile(x, y, z uint64) *Bound {
maxIndex := uint64(1) << z
if x < 0 || y < 0 || x >= maxIndex || y >= maxIndex {
panic("tile index out of range")
}
shift := 31 - z
if z > 31 {
shift = 0
}
lng1, lat1 := scalarMercatorInverse(x<<shift, y<<shift, 31)
lng2, lat2 := scalarMercatorInverse((x+1)<<shift, (y+1)<<shift, 31)
return &Bound{
sw: &Point{math.Min(lng1, lng2), math.Min(lat1, lat2)},
ne: &Point{math.Max(lng1, lng2), math.Max(lat1, lat2)},
}
}
// NewBoundFromGeoHash creates a new bound for the region defined by the GeoHash.
func NewBoundFromGeoHash(hash string) *Bound {
west, east, south, north := geoHash2ranges(hash)
return NewBound(west, east, south, north)
}
// NewBoundFromGeoHashInt64 creates a new bound from the region defined by the GeoHesh.
// bits indicates the precision of the hash.
func NewBoundFromGeoHashInt64(hash int64, bits int) *Bound {
west, east, south, north := geoHashInt2ranges(hash, bits)
return NewBound(west, east, south, north)
}
func geoHash2ranges(hash string) (float64, float64, float64, float64) {
latMin, latMax := -90.0, 90.0
lngMin, lngMax := -180.0, 180.0
even := true
for _, r := range hash {
// TODO: index step could probably be done better
i := strings.Index("0123456789bcdefghjkmnpqrstuvwxyz", string(r))
for j := 0x10; j != 0; j >>= 1 {
if even {
mid := (lngMin + lngMax) / 2.0
if i&j == 0 {
lngMax = mid
} else {
lngMin = mid
}
} else {
mid := (latMin + latMax) / 2.0
if i&j == 0 {
latMax = mid
} else {
latMin = mid
}
}
even = !even
}
}
return lngMin, lngMax, latMin, latMax
}
func geoBoundAroundPoint(center *Point, distance float64) *Bound {
radDist := distance / EarthRadius
radLat := deg2rad(center.Lat())
radLon := deg2rad(center.Lng())
minLat := radLat - radDist
maxLat := radLat + radDist
var minLon, maxLon float64
if minLat > minLatitude && maxLat < maxLatitude {
deltaLon := math.Asin(math.Sin(radDist) / math.Cos(radLat))
minLon = radLon - deltaLon
if minLon < minLongitude {
minLon += 2 * math.Pi
}
maxLon = radLon + deltaLon
if maxLon > maxLongitude {
maxLon -= 2 * math.Pi
}
} else {
minLat = math.Max(minLat, minLatitude)
maxLat = math.Min(maxLat, maxLatitude)
minLon = minLongitude
maxLon = maxLongitude
}
return &Bound{
sw: &Point{rad2deg(minLon), rad2deg(minLat)},
ne: &Point{rad2deg(maxLon), rad2deg(maxLat)},
}
}
func geoHashInt2ranges(hash int64, bits int) (float64, float64, float64, float64) {
latMin, latMax := -90.0, 90.0
lngMin, lngMax := -180.0, 180.0
var i int64
i = 1 << uint(bits)
for i != 0 {
i >>= 1
mid := (lngMin + lngMax) / 2.0
if hash&i == 0 {
lngMax = mid
} else {
lngMin = mid
}
i >>= 1
mid = (latMin + latMax) / 2.0
if hash&i == 0 {
latMax = mid
} else {
latMin = mid
}
}
return lngMin, lngMax, latMin, latMax
}
// Set allows for the modification of the bound values in place.
func (b *Bound) Set(west, east, south, north float64) {
b.sw[0] = west
b.sw[1] = south
b.ne[0] = east
b.ne[1] = north
}
// Extend grows the bound to include the new point.
func (b *Bound) Extend(point *Point) *Bound {
// already included, no big deal
if b.Contains(point) {
return b
}
b.sw.SetX(math.Min(b.sw.X(), point.X()))
b.ne.SetX(math.Max(b.ne.X(), point.X()))
b.sw.SetY(math.Min(b.sw.Y(), point.Y()))
b.ne.SetY(math.Max(b.ne.Y(), point.Y()))
return b
}
// Union extends this bounds to contain the union of this and the given bounds.
func (b *Bound) Union(other *Bound) *Bound {
b.Extend(other.SouthWest())
b.Extend(other.NorthWest())
b.Extend(other.SouthEast())
b.Extend(other.NorthEast())
return b
}
// Contains determines if the point is within the bound.
// Points on the boundary are considered within.
func (b *Bound) Contains(point *Point) bool {
if point.Y() < b.sw.Y() || b.ne.Y() < point.Y() {
return false
}
if point.X() < b.sw.X() || b.ne.X() < point.X() {
return false
}
return true
}
// Intersects determines if two bounds intersect.
// Returns true if they are touching.
func (b *Bound) Intersects(bound *Bound) bool {
if (b.ne[0] < bound.sw[0]) ||
(b.sw[0] > bound.ne[0]) ||
(b.ne[1] < bound.sw[1]) ||
(b.sw[1] > bound.ne[1]) {
return false
}
return true
}
// Center returns the center of the bound.
func (b *Bound) Center() *Point {
p := &Point{}
p.SetX((b.ne.X() + b.sw.X()) / 2.0)
p.SetY((b.ne.Y() + b.sw.Y()) / 2.0)
return p
}
// Pad expands the bound in all directions by the amount given. The amount must be
// in the units of the bounds. Technically one can pad with negative value,
// but no error checking is done.
func (b *Bound) Pad(amount float64) *Bound {
b.sw.SetX(b.sw.X() - amount)
b.sw.SetY(b.sw.Y() - amount)
b.ne.SetX(b.ne.X() + amount)
b.ne.SetY(b.ne.Y() + amount)
return b
}
// GeoPad expands the bound in all directions by the given amount of meters.
// Only applies if the data is Lng/Lat degrees.
func (b *Bound) GeoPad(meters float64) *Bound {
dy := meters / 111131.75
dx := dy / math.Cos(deg2rad(b.ne.Lat()))
dx = math.Max(dx, dy/math.Cos(deg2rad(b.sw.Lat())))
b.sw.SetLng(b.sw.Lng() - dx)
b.sw.SetLat(b.sw.Lat() - dy)
b.ne.SetLng(b.ne.Lng() + dx)
b.ne.SetLat(b.ne.Lat() + dy)
return b
}
// Height returns just the difference in the point's Y/Latitude.
func (b *Bound) Height() float64 {
return b.ne.Y() - b.sw.Y()
}
// Width returns just the difference in the point's X/Longitude.
func (b *Bound) Width() float64 {
return b.ne.X() - b.sw.X()
}
// GeoHeight returns the approximate height in meters.
// Only applies if the data is Lng/Lat degrees (EPSG:4326).
func (b *Bound) GeoHeight() float64 {
return 111131.75 * b.Height()
}
// GeoWidth returns the approximate width in meters
// of the center of the bound. Only applies if
// the data is Lng/Lat degrees.
func (b *Bound) GeoWidth(haversine ...bool) float64 {
c := b.Center()
A := &Point{b.sw[0], c[1]}
B := &Point{b.ne[0], c[1]}
return A.GeoDistanceFrom(B, yesHaversine(haversine))
}
// SouthWest returns the lower left corner of the bound.
func (b *Bound) SouthWest() *Point { return b.sw.Clone() }
// NorthEast returns the upper right corner of the bound.
func (b *Bound) NorthEast() *Point { return b.ne.Clone() }
// SouthEast returns the lower right corner of the bound.
func (b *Bound) SouthEast() *Point {
newP := &Point{}
newP.SetLat(b.sw.Lat()).SetLng(b.ne.Lng())
return newP
}
// NorthWest returns the upper left corner of the bound.
func (b *Bound) NorthWest() *Point {
newP := &Point{}
newP.SetLat(b.ne.Lat()).SetLng(b.sw.Lng())
return newP
}
// North returns the top of the bound.
func (b *Bound) North() float64 {
return b.ne[1]
}
// South returns the bottom of the bound.
func (b *Bound) South() float64 {
return b.sw[1]
}
// East returns the right of the bound.
func (b *Bound) East() float64 {
return b.ne[0]
}
// West returns the left of the bound.
func (b *Bound) West() float64 {
return b.sw[0]
}
// Top returns the north side of the bound.
func (b *Bound) Top() float64 {
return b.ne[1]
}
// Bottom returns the south side of the bound.
func (b *Bound) Bottom() float64 {
return b.sw[1]
}
// Right returns the east side of the bound.
func (b *Bound) Right() float64 {
return b.ne[0]
}
// Left returns the west side of the bound.
func (b *Bound) Left() float64 {
return b.sw[0]
}
// Empty returns true if it contains zero area or if
// it's in some malformed negative state where the left point is larger than the right.
// This can be caused by padding too much negative.
func (b *Bound) Empty() bool {
return b.sw.X() >= b.ne.X() || b.sw.Y() >= b.ne.Y()
}
// Equals returns if two bounds are equal.
func (b *Bound) Equals(c *Bound) bool {
if b.sw.Equals(c.sw) && b.ne.Equals(c.ne) {
return true
}
return false
}
// Clone returns a copy of the bound.
func (b *Bound) Clone() *Bound {
return NewBoundFromPoints(b.sw, b.ne)
}
// ToLine returns a Line from the southwest corner to the northeast.
func (b *Bound) ToLine() *Line {
return NewLine(b.sw, b.ne)
}
// String returns the string respentation of the bound in WKT format.
// POLYGON(west, south, west, north, east, north, east, south, west, south)
func (b *Bound) String() string {
// west, south, west, north, east, north, east, south, west, south
return fmt.Sprintf("POLYGON((%g %g, %g %g, %g %g, %g %g, %g %g))", b.sw[0], b.sw[1], b.sw[0], b.ne[1], b.ne[0], b.ne[1], b.ne[0], b.sw[1], b.sw[0], b.sw[1])
}
// ToMysqlIntersectsCondition returns a condition defining the intersection
// of the column and the bound. To be used in a MySQL query.
func (b *Bound) ToMysqlIntersectsCondition(column string) string {
return fmt.Sprintf("INTERSECTS(%s, GEOMFROMTEXT('%s'))", column, b.String())
}