forked from node-red/node-red-nodes
-
Notifications
You must be signed in to change notification settings - Fork 0
/
70-geohash.js
101 lines (97 loc) · 4.37 KB
/
70-geohash.js
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
/**
* Copyright 2014 IBM Corp.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
**/
module.exports = function(RED) {
"use strict";
var geohash = require('ngeohash');
function GeohashNode(n) {
RED.nodes.createNode(this,n);
var node = this;
var round = function(value, decimals) {
return Number(Math.round(value+'e'+decimals)+'e-'+decimals);
}
this.on("input", function(msg) {
if (msg.hasOwnProperty("location")) {
if (msg.location.hasOwnProperty("geohash")) {
var pos = geohash.decode(msg.location.geohash);
msg.location.lat = round(pos.latitude,5);
msg.location.lon = round(pos.longitude,5);
msg.location.error = { lat:round(pos.error.latitude,5), lon:round(pos.error.longitude,5) };
node.send(msg);
}
else {
var lt = msg.location.lat;
var ln = msg.location.lon;
var le = parseInt(msg.location.precision || 9);
if (le < 1) { le = 1; }
if (le > 9) { le = 9; }
if (lt && ln) {
msg.location.geohash = geohash.encode(lt, ln, le);
node.send(msg);
} else {
node.warn("lat or lon missing from msg.location");
}
}
}
else if (typeof msg.payload === "string") {
// try to decode it...
var regexp = new RegExp('^[a-z0-9]{1,9}$'); // can only contain a-z or 0-9 and length 1-9
if (regexp.test(msg.payload)) {
var po = geohash.decode(msg.payload);
msg.payload = { lat:round(po.latitude,5), lon:round(po.longitude,5) };
msg.payload.error = { lat:round(po.error.latitude,5), lon:round(po.error.longitude,5) };
node.send(msg);
}
else if (msg.payload.indexOf(",") !== -1) {
// has a comma so assume it's lat,lon(,precision)
var bits = msg.payload.split(",");
if (bits.length === 2) {
var la = Number(bits[0]);
if (la < -90) { la = -90; }
if (la > 90) { la = 90; }
var lo = Number(bits[1]);
if (lo < -180) { lo = ((lo-180)%360)+180; }
if (lo > 180) { lo = ((lo+180)%360)-180; }
if (!isNaN(la) && !isNaN(lo)) {
msg.payload = geohash.encode(la, lo);
node.send(msg);
} else {
node.warn("Incorrect string format - should be lat,lon");
}
}
else { node.warn("Unexpected string format - should be lat,lon"); }
}
else { node.warn("Unexpected string format - should either be lat,lon or geohash"); }
}
else if (typeof msg.payload === "object") {
var lat = msg.payload.lat || msg.payload.latitude;
var lon = msg.payload.lon || msg.payload.longitude;
var len = parseInt(msg.payload.precision || 9);
if (len < 1) { len = 1; }
if (len > 9) { len = 9; }
if (lat && lon) {
msg.payload.geohash = geohash.encode(lat, lon, len);
node.send(msg);
} else {
node.warn("lat or lon missing from msg.payload");
}
}
else {
node.warn("This node only expects strings or objects.");
}
});
}
RED.nodes.registerType("geohash",GeohashNode);
}