forked from paulmach/go.geo
-
Notifications
You must be signed in to change notification settings - Fork 0
/
point.go
340 lines (273 loc) · 8.61 KB
/
point.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
/*
Package geo is a geometry/geography library in Go. Its purpose is to allow for
basic point, line and path operations in the context of online mapping.
*/
package geo
import (
"fmt"
"math"
"strconv"
"github.com/paulmach/go.geojson"
)
// A Point is a simple X/Y or Lng/Lat 2d point. [X, Y] or [Lng, Lat]
type Point [2]float64
// InfinityPoint is the point at [inf, inf].
// Currently returned for the intersection of two collinear overlapping lines.
var InfinityPoint = &Point{math.Inf(1), math.Inf(1)}
// NewPoint creates a new point
func NewPoint(x, y float64) *Point {
return &Point{x, y}
}
// NewPointFromLatLng creates a new point from latlng
func NewPointFromLatLng(lat, lng float64) *Point {
return &Point{lng, lat}
}
// NewPointFromQuadkey creates a new point from a quadkey.
// See http://msdn.microsoft.com/en-us/library/bb259689.aspx for more information
// about this coordinate system.
func NewPointFromQuadkey(key int64, level int) *Point {
var x, y int64
var i uint
for i = 0; i < uint(level); i++ {
x |= (key & (1 << (2 * i))) >> i
y |= (key & (1 << (2*i + 1))) >> (i + 1)
}
lng, lat := scalarMercatorInverse(uint64(x), uint64(y), uint64(level))
return &Point{lng, lat}
}
// NewPointFromQuadkeyString creates a new point from a quadkey string.
func NewPointFromQuadkeyString(key string) *Point {
i, _ := strconv.ParseInt(key, 4, 64)
return NewPointFromQuadkey(i, len(key))
}
// NewPointFromGeoHash creates a new point at the center of the geohash range.
func NewPointFromGeoHash(hash string) *Point {
west, east, south, north := geoHash2ranges(hash)
return NewPoint((west+east)/2.0, (north+south)/2.0)
}
// NewPointFromGeoHashInt64 creates a new point at the center of the
// integer version of a geohash range. bits indicates the precision of the hash.
func NewPointFromGeoHashInt64(hash int64, bits int) *Point {
west, east, south, north := geoHashInt2ranges(hash, bits)
return NewPoint((west+east)/2.0, (north+south)/2.0)
}
// Point returns itself, so it implements the pointer interface.
func (p *Point) Point() *Point {
return p
}
// Transform applies a given projection or inverse projection to the current point.
func (p *Point) Transform(projector Projector) *Point {
projector(p)
return p
}
// DistanceFrom returns the Euclidean distance between the points.
func (p *Point) DistanceFrom(point *Point) float64 {
d0 := (point[0] - p[0])
d1 := (point[1] - p[1])
return math.Sqrt(d0*d0 + d1*d1)
}
// SquaredDistanceFrom returns the squared Euclidean distance between the points.
// This avoids a sqrt computation.
func (p *Point) SquaredDistanceFrom(point *Point) float64 {
d0 := (point[0] - p[0])
d1 := (point[1] - p[1])
return d0*d0 + d1*d1
}
// GeoDistanceFrom returns the geodesic distance in meters.
func (p *Point) GeoDistanceFrom(point *Point, haversine ...bool) float64 {
dLat := deg2rad(point.Lat() - p.Lat())
dLng := math.Abs(deg2rad(point.Lng() - p.Lng()))
if yesHaversine(haversine) {
// yes trig functions
dLat2Sin := math.Sin(dLat / 2)
dLng2Sin := math.Sin(dLng / 2)
a := dLat2Sin*dLat2Sin + math.Cos(deg2rad(p.Lat()))*math.Cos(deg2rad(point.Lat()))*dLng2Sin*dLng2Sin
return 2.0 * EarthRadius * math.Atan2(math.Sqrt(a), math.Sqrt(1-a))
}
if dLng > math.Pi {
dLng = 2*math.Pi - dLng
}
// fast way using pythagorean theorem on an equirectangular projection
x := dLng * math.Cos(deg2rad((p.Lat()+point.Lat())/2.0))
return math.Sqrt(dLat*dLat+x*x) * EarthRadius
}
// BearingTo computes the direction one must start traveling on earth
// to be heading to the given point.
func (p *Point) BearingTo(point *Point) float64 {
dLng := deg2rad(point.Lng() - p.Lng())
pLatRad := deg2rad(p.Lat())
pointLatRad := deg2rad(point.Lat())
y := math.Sin(dLng) * math.Cos(pointLatRad)
x := math.Cos(pLatRad)*math.Sin(pointLatRad) - math.Sin(pLatRad)*math.Cos(pointLatRad)*math.Cos(dLng)
return rad2deg(math.Atan2(y, x))
}
// Quadkey returns the quad key for the given point at the provided level.
// See http://msdn.microsoft.com/en-us/library/bb259689.aspx for more information
// about this coordinate system.
func (p *Point) Quadkey(level int) int64 {
x, y := scalarMercatorProject(p.Lng(), p.Lat(), uint64(level))
var i uint
var result uint64
for i = 0; i < uint(level); i++ {
result |= (x & (1 << i)) << i
result |= (y & (1 << i)) << (i + 1)
}
return int64(result)
}
// QuadkeyString returns the quad key for the given point at the provided level in string form
// See http://msdn.microsoft.com/en-us/library/bb259689.aspx for more information
// about this coordinate system.
func (p *Point) QuadkeyString(level int) string {
s := strconv.FormatInt(p.Quadkey(level), 4)
// for zero padding
zeros := "000000000000000000000000000000"
return zeros[:((level+1)-len(s))/2] + s
}
const base32 = "0123456789bcdefghjkmnpqrstuvwxyz"
// GeoHash returns the geohash string of a point representing a lng/lat location.
// The resulting hash will be `GeoHashPrecision` characters long, default is 12.
// Optionally one can include their required number of chars precision.
func (p *Point) GeoHash(chars ...int) string {
precision := GeoHashPrecision
if len(chars) > 0 {
precision = chars[0]
}
// 15 must be greater than GeoHashPrecision. If not, panic!!
var result [15]byte
hash := p.GeoHashInt64(5 * precision)
for i := 1; i <= precision; i++ {
result[precision-i] = byte(base32[hash&0x1F])
hash >>= 5
}
return string(result[:precision])
}
// GeoHashInt64 returns the integer version of the geohash
// down to the given number of bits.
// The main usecase for this function is to be able to do integer based ordering of points.
// In that case the number of bits should be the same for all encodings.
func (p *Point) GeoHashInt64(bits int) (hash int64) {
// This code was inspired by https://github.com/broady/gogeohash
latMin, latMax := -90.0, 90.0
lngMin, lngMax := -180.0, 180.0
for i := 0; i < bits; i++ {
hash <<= 1
// interleave bits
if i%2 == 0 {
mid := (lngMin + lngMax) / 2.0
if p[0] > mid {
lngMin = mid
hash |= 1
} else {
lngMax = mid
}
} else {
mid := (latMin + latMax) / 2.0
if p[1] > mid {
latMin = mid
hash |= 1
} else {
latMax = mid
}
}
}
return
}
// Add a point to the given point.
func (p *Point) Add(point *Point) *Point {
p[0] += point[0]
p[1] += point[1]
return p
}
// Subtract a point from the given point.
func (p *Point) Subtract(point *Point) *Point {
p[0] -= point[0]
p[1] -= point[1]
return p
}
// Normalize treats the point as a vector and
// scales it such that its distance from [0,0] is 1.
func (p *Point) Normalize() *Point {
dist := math.Sqrt(p[0]*p[0] + p[1]*p[1])
if dist == 0 {
p[0] = 0
p[1] = 0
return p
}
p[0] /= dist
p[1] /= dist
return p
}
// Scale each component of the point.
func (p *Point) Scale(factor float64) *Point {
p[0] *= factor
p[1] *= factor
return p
}
// Dot is just x1*x2 + y1*y2
func (p *Point) Dot(v *Point) float64 {
return p[0]*v[0] + p[1]*v[1]
}
// ToArray casts the data to a [2]float64.
func (p Point) ToArray() [2]float64 {
return [2]float64(p)
}
// Clone creates a duplicate of the point.
func (p Point) Clone() *Point {
return &p
}
// Equals checks if the point represents the same point or vector.
func (p *Point) Equals(point *Point) bool {
if p[0] == point[0] && p[1] == point[1] {
return true
}
return false
}
// Lat returns the latitude/vertical component of the point.
func (p *Point) Lat() float64 {
return p[1]
}
// SetLat sets the latitude/vertical component of the point.
func (p *Point) SetLat(lat float64) *Point {
p[1] = lat
return p
}
// Lng returns the longitude/horizontal component of the point.
func (p *Point) Lng() float64 {
return p[0]
}
// SetLng sets the longitude/horizontal component of the point.
func (p *Point) SetLng(lng float64) *Point {
p[0] = lng
return p
}
// X returns the x/horizontal component of the point.
func (p *Point) X() float64 {
return p[0]
}
// SetX sets the x/horizontal component of the point.
func (p *Point) SetX(x float64) *Point {
p[0] = x
return p
}
// Y returns the y/vertical component of the point.
func (p *Point) Y() float64 {
return p[1]
}
// SetY sets the y/vertical component of the point.
func (p *Point) SetY(y float64) *Point {
p[1] = y
return p
}
// ToGeoJSON creates a new geojson feature with a point geometry.
func (p Point) ToGeoJSON() *geojson.Feature {
return geojson.NewPointFeature([]float64{p[0], p[1]})
}
// ToWKT returns the point in WKT format, eg. POINT(30.5 10.5)
func (p Point) ToWKT() string {
return p.String()
}
// String returns a string representation of the point.
// The format is WKT, e.g. POINT(30.5 10.5)
func (p Point) String() string {
return fmt.Sprintf("POINT(%g %g)", p[0], p[1])
}