@idm-plugin/geo 1.2.2 → 1.2.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,2 +1,2 @@
1
- ## Geo Plugin
1
+ ## Geo Plugin
2
2
  Geo插件
package/dist/index.js CHANGED
@@ -1,11 +1,11 @@
1
1
  import * as g from "@turf/turf";
2
- import N from "moment";
2
+ import D from "moment";
3
3
  import "moment-timezone";
4
- import E from "tz-lookup";
5
- function F(C) {
4
+ import W from "tz-lookup";
5
+ function E(C) {
6
6
  return C && C.__esModule && Object.prototype.hasOwnProperty.call(C, "default") ? C.default : C;
7
7
  }
8
- class x {
8
+ class R {
9
9
  log() {
10
10
  }
11
11
  isLevelEnabled() {
@@ -19,20 +19,20 @@ class x {
19
19
  }
20
20
  }
21
21
  ["Trace", "Debug", "Info", "Warn", "Error", "Fatal", "Mark"].forEach((C) => {
22
- x.prototype[C.toLowerCase()] = () => {
23
- }, x.prototype[`is${C}Enabled`] = () => !1;
22
+ R.prototype[C.toLowerCase()] = () => {
23
+ }, R.prototype[`is${C}Enabled`] = () => !1;
24
24
  });
25
- const L = () => {
25
+ const j = () => {
26
26
  try {
27
27
  return require("log4js");
28
28
  } catch {
29
29
  return null;
30
30
  }
31
- }, W = L(), k = W ? W.getLogger : () => new x();
31
+ }, y = j(), k = y ? y.getLogger : () => new R();
32
32
  var A = {
33
33
  getLogger: k
34
34
  };
35
- const j = /* @__PURE__ */ F(A);
35
+ const F = /* @__PURE__ */ E(A);
36
36
  class l {
37
37
  /**
38
38
  * 基于输入的经度,计算出时区
@@ -40,8 +40,8 @@ class l {
40
40
  * @param lat
41
41
  */
42
42
  static guessTimeZoneOffset(e, t) {
43
- const n = E(t, e), o = N().tz(n).utcOffset();
44
- return this.roundPrecision(o / 60, 1);
43
+ const n = W(t, e), i = D().tz(n).utcOffset();
44
+ return this.roundPrecision(i / 60, 1);
45
45
  }
46
46
  /**
47
47
  * 将时间offset转换为时区,例如:8.5 => +08:30
@@ -54,13 +54,13 @@ class l {
54
54
  }
55
55
  static lng2pretty(e, t = 6, n = "H°M′") {
56
56
  e = l.convertToStdLng(e, t);
57
- let o = "E";
58
- e < 0 && (o = "W"), e = Math.abs(e), n = n.toUpperCase();
59
- let i = e * 3600, s, r, c, a, d, u;
60
- s = i % 3600 % 60, n.indexOf("S") !== -1 && (i = i - s, r = l.padNumber(s, 2, 2)), c = i / 60 % 60, n.indexOf("M") !== -1 && (n.indexOf("S") !== -1 ? a = l.roundPrecision(c, t).toString().padStart(2, "0") : a = l.padNumber(c, 2, 2), i = i - c * 60), d = i / 3600, n.indexOf("M") !== -1 ? u = l.roundPrecision(d, t).toString().padStart(3, "0") : u = l.padNumber(d, 3, 2);
61
- const p = `${n.replace(/S+/gi, r).replace(/M+/gi, a).replace(/H+/gi, u)}${o}`;
57
+ let i = "E";
58
+ e < 0 && (i = "W"), e = Math.abs(e), n = n.toUpperCase();
59
+ let o = e * 3600, s, r, c, a, d, u;
60
+ s = o % 3600 % 60, n.indexOf("S") !== -1 && (o = o - s, r = l.padNumber(s, 2, 2)), c = o / 60 % 60, n.indexOf("M") !== -1 && (n.indexOf("S") !== -1 ? a = l.roundPrecision(c, t).toString().padStart(2, "0") : a = l.padNumber(c, 2, 2), o = o - c * 60), d = o / 3600, n.indexOf("M") !== -1 ? u = l.roundPrecision(d, t).toString().padStart(3, "0") : u = l.padNumber(d, 3, 2);
61
+ const p = `${n.replace(/S+/gi, r).replace(/M+/gi, a).replace(/H+/gi, u)}${i}`;
62
62
  return {
63
- direction: o,
63
+ direction: i,
64
64
  degree: l.roundPrecision(d, t),
65
65
  minute: l.roundPrecision(c, t),
66
66
  second: l.roundPrecision(s, t),
@@ -75,13 +75,13 @@ class l {
75
75
  */
76
76
  static lat2pretty(e, t = 6, n = "H°M′") {
77
77
  e = e % 180;
78
- let o = "N";
79
- e < 0 && (o = "S"), e = Math.abs(e), n = n.toUpperCase();
80
- let i = e * 3600, s, r, c, a, d, u;
81
- s = i % 3600 % 60, n.indexOf("S") !== -1 && (i = i - s, r = l.padNumber(s, 2, 2)), c = i / 60 % 60, n.indexOf("M") !== -1 && (n.indexOf("S") !== -1 ? a = l.roundPrecision(c, t).toString().padStart(2, "0") : a = l.padNumber(c, 2, 2), i = i - c * 60), d = i / 3600, n.indexOf("M") !== -1 ? u = l.roundPrecision(d, t).toString().padStart(2, "0") : u = l.padNumber(d, 2, 2);
82
- const p = `${n.replace(/S+/gi, r).replace(/M+/gi, a).replace(/H+/gi, u)}${o}`;
78
+ let i = "N";
79
+ e < 0 && (i = "S"), e = Math.abs(e), n = n.toUpperCase();
80
+ let o = e * 3600, s, r, c, a, d, u;
81
+ s = o % 3600 % 60, n.indexOf("S") !== -1 && (o = o - s, r = l.padNumber(s, 2, 2)), c = o / 60 % 60, n.indexOf("M") !== -1 && (n.indexOf("S") !== -1 ? a = l.roundPrecision(c, t).toString().padStart(2, "0") : a = l.padNumber(c, 2, 2), o = o - c * 60), d = o / 3600, n.indexOf("M") !== -1 ? u = l.roundPrecision(d, t).toString().padStart(2, "0") : u = l.padNumber(d, 2, 2);
82
+ const p = `${n.replace(/S+/gi, r).replace(/M+/gi, a).replace(/H+/gi, u)}${i}`;
83
83
  return {
84
- direction: o,
84
+ direction: i,
85
85
  degree: l.roundPrecision(d, t),
86
86
  minute: l.roundPrecision(c, t),
87
87
  second: l.roundPrecision(s, t),
@@ -92,15 +92,15 @@ class l {
92
92
  let n;
93
93
  if (isNaN(e)) {
94
94
  e = l.strReplace(e, "LNG");
95
- const o = e[e.length - 1].toUpperCase();
95
+ const i = e[e.length - 1].toUpperCase();
96
96
  e = e.substring(0, e.length - 1).trim();
97
- const i = e.split(" ").filter((c) => c !== "").map((c) => Number(c));
98
- let [s, r] = i;
97
+ const o = e.split(" ").filter((c) => c !== "").map((c) => Number(c));
98
+ let [s, r] = o;
99
99
  if (r = r > 60 ? r / Math.pow(10, String(r).length - 2) : r, s > 360 && !r) {
100
100
  const c = this.roundPrecision(s / 100, 0);
101
101
  r = s - c * 100, s = c;
102
102
  }
103
- n = s + (r ?? 0) / 60, o === "W" && (n = n * -1);
103
+ n = s + (r ?? 0) / 60, i === "W" && (n = n * -1);
104
104
  } else
105
105
  n = Number(e);
106
106
  return l.convertToStdLng(n, t);
@@ -109,23 +109,23 @@ class l {
109
109
  let n;
110
110
  if (isNaN(e)) {
111
111
  e = l.strReplace(e, "LAT");
112
- const o = e[e.length - 1].toUpperCase();
112
+ const i = e[e.length - 1].toUpperCase();
113
113
  e = e.substring(0, e.length - 1).trim();
114
- const i = e.split(" ").filter((c) => c !== "").map((c) => Number(c));
115
- let [s, r] = i;
114
+ const o = e.split(" ").filter((c) => c !== "").map((c) => Number(c));
115
+ let [s, r] = o;
116
116
  if (r = r > 60 ? r / Math.pow(10, String(r).length - 2) : r, s > 90 && !r) {
117
117
  const c = this.roundPrecision(s / 100, 0);
118
118
  r = s - c * 100, s = c;
119
119
  }
120
- n = s + (r ?? 0) / 60, o === "S" && (n = n * -1);
120
+ n = s + (r ?? 0) / 60, i === "S" && (n = n * -1);
121
121
  } else
122
122
  n = Number(e);
123
123
  return l.roundPrecision(n, t);
124
124
  }
125
125
  static str2LngOrLat(e, t = 6, n = "LAT") {
126
126
  e = l.strReplace(e, n);
127
- const o = e[e.length - 1].toUpperCase();
128
- return ["N", "S"].includes(o) ? {
127
+ const i = e[e.length - 1].toUpperCase();
128
+ return ["N", "S"].includes(i) ? {
129
129
  lat: l.str2Lat(e, t)
130
130
  } : {
131
131
  lng: l.str2Lng(e, t)
@@ -163,10 +163,10 @@ class l {
163
163
  e = e.replace(/([0-9]+)\.([0-9]+\.[0-9]+)/g, "$1 $2").replace(/-/g, " ").replace(/°/, " ").replace(/'/g, " ").replace(/′/g, " ").replace(/"/g, " ").replace(/∼/g, " ").replace(/°/g, " ").replace(/,/g, ".").replace(/^ /g, "").replace(/ $/g, "").trim();
164
164
  const n = e[e.length - 1].toUpperCase();
165
165
  if (!["N", "S", "E", "W"].includes(n)) {
166
- const o = e, i = Number(o.split(" ")[0]);
167
- if (isNaN(i))
166
+ const i = e, o = Number(i.split(" ")[0]);
167
+ if (isNaN(o))
168
168
  throw new Error(`invalid Lat/Lng: ${e}`);
169
- i >= 90 ? e = `${o}E` : i <= -90 ? e = `${o}W` : ["LAN", "LNG"].includes(t == null ? void 0 : t.toUpperCase()) ? e = `${o}${i > 0 ? "E" : "W"}` : e = `${o}${i > 0 ? "N" : "S"}`;
169
+ o >= 90 ? e = `${i}E` : o <= -90 ? e = `${i}W` : ["LAN", "LNG"].includes(t == null ? void 0 : t.toUpperCase()) ? e = `${i}${o > 0 ? "E" : "W"}` : e = `${i}${o > 0 ? "N" : "S"}`;
170
170
  }
171
171
  return e;
172
172
  }
@@ -178,11 +178,11 @@ class l {
178
178
  * @param dcmPrecision 小数位数
179
179
  */
180
180
  static padNumber(e, t = 2, n = 2) {
181
- const o = Math.trunc(e).toString().padStart(t, "0"), i = Math.trunc(l.roundPrecision(e - Math.trunc(e), n) * Math.pow(10, n)).toString().padStart(n, "0");
182
- return `${o}.${i}`;
181
+ const i = Math.trunc(e).toString().padStart(t, "0"), o = Math.trunc(l.roundPrecision(e - Math.trunc(e), n) * Math.pow(10, n)).toString().padStart(n, "0");
182
+ return `${i}.${o}`;
183
183
  }
184
184
  }
185
- class S {
185
+ class P {
186
186
  /**
187
187
  * 计算方位角
188
188
  * @param from 坐标 {lng, lat}
@@ -191,13 +191,13 @@ class S {
191
191
  * @param precision
192
192
  * @returns {number} 单位度
193
193
  */
194
- static calculateBearing(e, t, n = !0, o = 4) {
195
- const i = g.points([
194
+ static calculateBearing(e, t, n = !0, i = 4) {
195
+ const o = g.points([
196
196
  [e.lng, e.lat],
197
197
  [t.lng, t.lat]
198
198
  ]);
199
199
  let s;
200
- return n ? s = g.rhumbBearing(i.features[0], i.features[1]) : s = g.bearing(i.features[0], i.features[1]), s < 0 && (s += 360), l.roundPrecision(s, o);
200
+ return n ? s = g.rhumbBearing(o.features[0], o.features[1]) : s = g.bearing(o.features[0], o.features[1]), s < 0 && (s += 360), l.roundPrecision(s, i);
201
201
  }
202
202
  /**
203
203
  * 计算两点间距离
@@ -208,14 +208,14 @@ class S {
208
208
  * @param units 单位,默认 nm(海里)
209
209
  * @returns {number}
210
210
  */
211
- static calculateDistance(e, t, n = !0, o = 4, i = "nauticalmiles") {
212
- e = { ...e }, t = { ...t }, e.lng = l.convertToStdLng(e.lng, o), t.lng = l.convertToStdLng(t.lng, o);
211
+ static calculateDistance(e, t, n = !0, i = 4, o = "nauticalmiles") {
212
+ e = { ...e }, t = { ...t }, e.lng = l.convertToStdLng(e.lng, i), t.lng = l.convertToStdLng(t.lng, i);
213
213
  const s = g.points([
214
214
  [e.lng, e.lat],
215
215
  [t.lng, t.lat]
216
216
  ]);
217
217
  let r;
218
- return n ? r = g.rhumbDistance(s.features[0], s.features[1], { units: i }) : r = g.distance(s.features[0], s.features[1], { units: i }), l.roundPrecision(r, o);
218
+ return n ? r = g.rhumbDistance(s.features[0], s.features[1], { units: o }) : r = g.distance(s.features[0], s.features[1], { units: o }), l.roundPrecision(r, i);
219
219
  }
220
220
  /**
221
221
  * 计算航线距离
@@ -224,15 +224,15 @@ class S {
224
224
  * @param units
225
225
  */
226
226
  static calculateRouteDistance(e, t = 4, n = "nauticalmiles") {
227
- let o = 0, i;
227
+ let i = 0, o;
228
228
  for (const s of e)
229
229
  for (let r = 0; r < s.length - 1; r++) {
230
230
  const c = { lng: s[r][0], lat: s[r][1] };
231
- r === 0 && i && (o += this.calculateDistance(i, c, !0, t, n));
231
+ r === 0 && o && (i += this.calculateDistance(o, c, !0, t, n));
232
232
  const a = { lng: s[r + 1][0], lat: s[r + 1][1] };
233
- o += this.calculateDistance(c, a, !0, t, n), i = a;
233
+ i += this.calculateDistance(c, a, !0, t, n), o = a;
234
234
  }
235
- return l.roundPrecision(o, t);
235
+ return l.roundPrecision(i, t);
236
236
  }
237
237
  /**
238
238
  * 计算坐标(基于方位角和距离)
@@ -242,10 +242,10 @@ class S {
242
242
  * @param units 单位,默认 nm(海里)
243
243
  * @param rhumb
244
244
  */
245
- static calculateCoordinate(e, t, n, o = "nauticalmiles", i = !0) {
245
+ static calculateCoordinate(e, t, n, i = "nauticalmiles", o = !0) {
246
246
  const s = g.point([e.lng, e.lat]);
247
247
  let r;
248
- i ? r = g.rhumbDestination(s, n, t, { units: o }) : r = g.destination(s, n, t, { units: o });
248
+ o ? r = g.rhumbDestination(s, n, t, { units: i }) : r = g.destination(s, n, t, { units: i });
249
249
  const c = r.geometry.coordinates;
250
250
  return { lng: l.convertToStdLng(c[0], 8), lat: l.roundPrecision(c[1], 8) };
251
251
  }
@@ -258,13 +258,13 @@ class S {
258
258
  * @param includeTail true 包含终点 to
259
259
  * @param units 单位,默认 nm(海里)
260
260
  */
261
- static interpolateCoordinates(e, t, n, o = !0, i = !0, s = "nauticalmiles") {
261
+ static interpolateCoordinates(e, t, n, i = !0, o = !0, s = "nauticalmiles") {
262
262
  const r = [], c = this.calculateBearing(e, t, !1), a = this.calculateDistance(e, t, !1, 8, s);
263
- o && r.push({ lng: e.lng, lat: e.lat });
263
+ i && r.push({ lng: e.lng, lat: e.lat });
264
264
  let d = 0;
265
265
  for (; d < a; )
266
266
  d += n, d < a && r.push(this.calculateCoordinate(e, c, d, s, !1));
267
- return i && r.push({ lng: t.lng, lat: t.lat }), r;
267
+ return o && r.push({ lng: t.lng, lat: t.lat }), r;
268
268
  }
269
269
  /**
270
270
  * 分组坐标(如相邻两个坐标经度差超180度,需以180为界将坐标分为两组)
@@ -282,14 +282,14 @@ class S {
282
282
  return [];
283
283
  e = this.deduplicateCoordinates(e);
284
284
  let n = [];
285
- const o = [];
286
- let i, s;
285
+ const i = [];
286
+ let o, s;
287
287
  for (let r = 0; r < e.length - 1; r++) {
288
- i = l.convertToStdLng(e[r].lng, 8), s = l.convertToStdLng(e[r + 1].lng, 8), n.push([i, e[r].lat]);
289
- const c = i - s;
288
+ o = l.convertToStdLng(e[r].lng, 8), s = l.convertToStdLng(e[r + 1].lng, 8), n.push([o, e[r].lat]);
289
+ const c = o - s;
290
290
  if (Math.abs(c) > 180) {
291
291
  const a = l.convertToMonotonicLng2([
292
- [i, e[r].lat],
292
+ [o, e[r].lat],
293
293
  [s, e[r + 1].lat]
294
294
  ]);
295
295
  let d, u;
@@ -298,17 +298,17 @@ class S {
298
298
  [c > 0 ? 180 : -180, -89]
299
299
  ])) : (d = g.greatCircle(a[0], a[1]), u = g.greatCircle([c > 0 ? 180 : -180, 89], [c > 0 ? 180 : -180, -89]));
300
300
  const p = g.lineIntersect(d, u);
301
- let f;
301
+ let h;
302
302
  if (p.features.length) {
303
- const M = g.getCoord(p.features[0]);
304
- f = l.roundPrecision(M[1], 8);
303
+ const S = g.getCoord(p.features[0]);
304
+ h = l.roundPrecision(S[1], 8);
305
305
  } else
306
- f = e[r].lat;
307
- c > 0 ? (n.push([180 - 1e-6, f]), o.push([...n]), n = [], n.push([-(180 - 1e-6), f])) : (n.push([-(180 - 1e-6), f]), o.push([...n]), n = [], n.push([180 - 1e-6, f]));
306
+ h = e[r].lat;
307
+ c > 0 ? (n.push([180 - 1e-6, h]), i.push([...n]), n = [], n.push([-(180 - 1e-6), h])) : (n.push([-(180 - 1e-6), h]), i.push([...n]), n = [], n.push([180 - 1e-6, h]));
308
308
  }
309
309
  r === e.length - 2 && n.push([s, e[r + 1].lat]);
310
310
  }
311
- return o.push(n), o;
311
+ return i.push(n), i;
312
312
  }
313
313
  /**
314
314
  * 去除重复坐标
@@ -317,8 +317,8 @@ class S {
317
317
  static deduplicateRoute(e) {
318
318
  const t = [];
319
319
  for (const n of e) {
320
- const o = n.reduce((i, s) => (i.findIndex((r) => r[0] === s[0] && r[1] === s[1]) === -1 && i.push(s), i), []);
321
- t.push(o);
320
+ const i = n.reduce((o, s) => (o.findIndex((r) => r[0] === s[0] && r[1] === s[1]) === -1 && o.push(s), o), []);
321
+ t.push(i);
322
322
  }
323
323
  return t;
324
324
  }
@@ -327,7 +327,7 @@ class S {
327
327
  * @param coordinates
328
328
  */
329
329
  static deduplicateCoordinates(e) {
330
- return e.reduce((t, n) => (t.findIndex((o) => o.lat === n.lat && o.lng === n.lng) === -1 && t.push(n), t), []);
330
+ return e.reduce((t, n) => (t.findIndex((i) => i.lat === n.lat && i.lng === n.lng) === -1 && t.push(n), t), []);
331
331
  }
332
332
  /**
333
333
  * 移出坐标
@@ -337,8 +337,8 @@ class S {
337
337
  static removeCoordinateFromRoute(e, t) {
338
338
  e.lng = l.convertToStdLng(e.lng, 8);
339
339
  for (const n of t)
340
- for (let o = n.length - 1; o >= 0; o--)
341
- l.roundPrecision(n[o][0], 8) === e.lng && l.roundPrecision(n[o][1], 8) === l.roundPrecision(e.lat, 8) && n.splice(o, 1);
340
+ for (let i = n.length - 1; i >= 0; i--)
341
+ l.roundPrecision(n[i][0], 8) === e.lng && l.roundPrecision(n[i][1], 8) === l.roundPrecision(e.lat, 8) && n.splice(i, 1);
342
342
  return t;
343
343
  }
344
344
  /**
@@ -365,13 +365,13 @@ class S {
365
365
  */
366
366
  static mergeCoordinateToRoute(e, t) {
367
367
  e.lng = l.convertToStdLng(e.lng, 8);
368
- let n = Number.MAX_VALUE, o = 0, i = 0, s, r;
368
+ let n = Number.MAX_VALUE, i = 0, o = 0, s, r;
369
369
  return t.forEach((c, a) => {
370
370
  for (let d = 0; d < c.length - 1; d++) {
371
- const u = { lng: c[d][0], lat: c[d][1] }, p = { lng: c[d + 1][0], lat: c[d + 1][1] }, f = this.calculatePointToLineDistance(e, u, p);
372
- n > f && (n = f, i = d, o = a, s = this.calculateDistance(u, e), r = this.calculateDistance(p, e));
371
+ const u = { lng: c[d][0], lat: c[d][1] }, p = { lng: c[d + 1][0], lat: c[d + 1][1] }, h = this.calculatePointToLineDistance(e, u, p);
372
+ n > h && (n = h, o = d, i = a, s = this.calculateDistance(u, e), r = this.calculateDistance(p, e));
373
373
  }
374
- }), s !== 0 && r !== 0 ? t[o].splice(i + 1, 0, [e.lng, e.lat]) : s === 0 ? t[o].splice(i, 1, [e.lng, e.lat]) : r === 0 && t[o].splice(i + 1, 1, [e.lng, e.lat]), t;
374
+ }), s !== 0 && r !== 0 ? t[i].splice(o + 1, 0, [e.lng, e.lat]) : s === 0 ? t[i].splice(o, 1, [e.lng, e.lat]) : r === 0 && t[i].splice(o + 1, 1, [e.lng, e.lat]), t;
375
375
  }
376
376
  /**
377
377
  * 向Route尾加1个坐标
@@ -380,8 +380,8 @@ class S {
380
380
  */
381
381
  static appendCoordinateToRoute(e, t) {
382
382
  e.lng = l.convertToStdLng(e.lng, 8);
383
- const n = S.convertRouteToCoordinates(t);
384
- return n.push(e), S.divideAccordingToLng(n);
383
+ const n = P.convertRouteToCoordinates(t);
384
+ return n.push(e), P.divideAccordingToLng(n);
385
385
  }
386
386
  /**
387
387
  * 向route头加1个坐标
@@ -389,8 +389,8 @@ class S {
389
389
  * @param route
390
390
  */
391
391
  static unshiftCoordinateToRoute(e, t) {
392
- const n = S.convertRouteToCoordinates(t);
393
- return n.unshift(e), S.divideAccordingToLng(n);
392
+ const n = P.convertRouteToCoordinates(t);
393
+ return n.unshift(e), P.divideAccordingToLng(n);
394
394
  }
395
395
  /**
396
396
  * 合并多个waypoints进航线
@@ -411,21 +411,21 @@ class S {
411
411
  */
412
412
  static calculateRangeRoute(e, t, n) {
413
413
  n = this.mergeWaypointsToRoute([e, t], n);
414
- const o = [];
415
- let i = 0;
414
+ const i = [];
415
+ let o = 0;
416
416
  return n.forEach((s) => {
417
- if (i === 2)
417
+ if (o === 2)
418
418
  return;
419
419
  const r = [];
420
420
  for (const c of s) {
421
421
  if (l.roundPrecision(t.lng, 8) === l.roundPrecision(c[0], 8) && l.roundPrecision(t.lat, 8) === l.roundPrecision(c[1], 8)) {
422
- r.push(c), i === 0 && r.push([e.lng, e.lat]), i = 2;
422
+ r.push(c), o === 0 && r.push([e.lng, e.lat]), o = 2;
423
423
  break;
424
424
  }
425
- i === 1 ? r.push(c) : l.roundPrecision(e.lng, 8) === l.roundPrecision(c[0], 8) && l.roundPrecision(e.lat, 8) === l.roundPrecision(c[1], 8) && (i = 1, r.push(c));
425
+ o === 1 ? r.push(c) : l.roundPrecision(e.lng, 8) === l.roundPrecision(c[0], 8) && l.roundPrecision(e.lat, 8) === l.roundPrecision(c[1], 8) && (o = 1, r.push(c));
426
426
  }
427
- r.length && o.push(r);
428
- }), o;
427
+ r.length && i.push(r);
428
+ }), i;
429
429
  }
430
430
  /**
431
431
  * 计算from到to之间的航线
@@ -435,8 +435,8 @@ class S {
435
435
  * @param waypoints
436
436
  * @return [{lng, lat}]
437
437
  */
438
- static calculateRangeWaypoints(e, t, n, o = []) {
439
- const i = this.convertRouteToCoordinates(n, 0), s = this.mergeCoordinatesToWaypoints([e, t, ...o], i), r = s.findIndex(
438
+ static calculateRangeWaypoints(e, t, n, i = []) {
439
+ const o = this.convertRouteToCoordinates(n, 0), s = this.mergeCoordinatesToWaypoints([e, t, ...i], o), r = s.findIndex(
440
440
  (d) => l.roundPrecision(e.lng, 8) === l.roundPrecision(d.lng, 8) && l.roundPrecision(e.lat, 8) === l.roundPrecision(d.lat, 8)
441
441
  ), c = s.findIndex(
442
442
  (d) => l.roundPrecision(t.lng, 8) === l.roundPrecision(d.lng, 8) && l.roundPrecision(t.lat, 8) === l.roundPrecision(d.lat, 8)
@@ -449,13 +449,13 @@ class S {
449
449
  * @param route
450
450
  */
451
451
  static calculateMinDistanceToRoute(e, t) {
452
- let n = Number.MAX_VALUE, o = 0, i = 0;
452
+ let n = Number.MAX_VALUE, i = 0, o = 0;
453
453
  return t.forEach((s, r) => {
454
454
  for (let c = 0; c < s.length - 1; c++) {
455
455
  const a = { lng: s[c][0], lat: s[c][1] }, d = { lng: s[c + 1][0], lat: s[c + 1][1] }, u = this.calculatePointToLineDistance(e, a, d);
456
- n > u && (n = u, o = c, i = r);
456
+ n > u && (n = u, i = c, o = r);
457
457
  }
458
- }), { minDist: n, segIndex: i, minIndex: o };
458
+ }), { minDist: n, segIndex: o, minIndex: i };
459
459
  }
460
460
  /**
461
461
  * 计算子航线
@@ -464,20 +464,20 @@ class S {
464
464
  * @return [[[lng, lat]]]
465
465
  */
466
466
  static calculateSubRoute(e, t) {
467
- const { segIndex: n, minIndex: o } = this.calculateMinDistanceToRoute({ ...e }, t);
467
+ const { segIndex: n, minIndex: i } = this.calculateMinDistanceToRoute({ ...e }, t);
468
468
  e.lng = l.convertToStdLng(e.lng);
469
- const i = [];
469
+ const o = [];
470
470
  let s = !0;
471
471
  for (let r = n; r < t.length; r++)
472
472
  if (s) {
473
473
  const c = [];
474
474
  c.push([e.lng, e.lat]);
475
- for (let a = o + 1; a < t[r].length; a++)
475
+ for (let a = i + 1; a < t[r].length; a++)
476
476
  e.lng === t[r][a][0] && e.lat === t[r][a][1] || c.push(t[r][a]);
477
- i.push(c), s = !1;
477
+ o.push(c), s = !1;
478
478
  } else
479
- i.push([...t[r]]);
480
- return i;
479
+ o.push([...t[r]]);
480
+ return o;
481
481
  }
482
482
  /**
483
483
  * 计算子途经点
@@ -486,7 +486,7 @@ class S {
486
486
  * @return [{lng, lat}]
487
487
  */
488
488
  static calculateSubWaypoints(e, t) {
489
- let n = Number.MAX_VALUE, o = 0;
489
+ let n = Number.MAX_VALUE, i = 0;
490
490
  for (let s = 0; s < t.length - 1; s++) {
491
491
  const r = t[s], c = t[s + 1];
492
492
  if (this.calculateDistance(e, r) === 0)
@@ -494,13 +494,13 @@ class S {
494
494
  if (this.calculateDistance(e, c) === 0)
495
495
  return t.filter((d, u) => u > 0);
496
496
  const a = this.calculatePointToLineDistance(e, r, c);
497
- n > a && (n = a, o = s);
497
+ n > a && (n = a, i = s);
498
498
  }
499
499
  e.lng = l.convertToStdLng(e.lng);
500
- const i = [e];
501
- for (let s = o + 1; s < t.length; s++)
502
- i.push(t[s]);
503
- return i;
500
+ const o = [e];
501
+ for (let s = i + 1; s < t.length; s++)
502
+ o.push(t[s]);
503
+ return o;
504
504
  }
505
505
  /**
506
506
  * 计算坐标到以(from, to)横向线上的距离
@@ -509,14 +509,14 @@ class S {
509
509
  * @param to { lng, lat }
510
510
  * @param options
511
511
  */
512
- static calculatePointToLineDistance(e, t, n, o = { units: "nauticalmiles", method: "geodesic" }) {
512
+ static calculatePointToLineDistance(e, t, n, i = { units: "nauticalmiles", method: "geodesic" }) {
513
513
  e.lng = l.convertToStdLng(e.lng), t = { ...t }, n = { ...n }, t.lng = l.convertToStdLng(t.lng, 8), n.lng = l.convertToStdLng(n.lng, 8);
514
- const i = l.convertToMonotonicLng([t, n]);
515
- t = i[0], n = i[1];
514
+ const o = l.convertToMonotonicLng([t, n]);
515
+ t = o[0], n = o[1];
516
516
  const s = g.lineString([
517
517
  [t.lng, t.lat],
518
518
  [n.lng, n.lat]
519
- ]), r = g.pointToLineDistance(g.point([e.lng, e.lat]), s, o), c = g.pointToLineDistance(g.point([e.lng > 0 ? e.lng - 360 : e.lng + 360, e.lat]), s, o);
519
+ ]), r = g.pointToLineDistance(g.point([e.lng, e.lat]), s, i), c = g.pointToLineDistance(g.point([e.lng > 0 ? e.lng - 360 : e.lng + 360, e.lat]), s, i);
520
520
  return l.roundPrecision(Math.min(r, c), 6);
521
521
  }
522
522
  /**
@@ -527,8 +527,8 @@ class S {
527
527
  static calculateWaypointsPropInRoute(e, t) {
528
528
  t = this.mergeWaypointsToRoute(e, t);
529
529
  for (let n = 0; n < e.length - 1; n++) {
530
- const o = e[n], i = e[n + 1], s = this.calculateRangeRoute(o, i, t);
531
- n === 0 && (o.distanceFromPrevious = 0, o.distanceFromStart = 0), i.distanceFromPrevious = this.calculateRouteDistance(s), i.distanceFromStart = l.roundPrecision((o.distanceFromStart || 0) + i.distanceFromPrevious);
530
+ const i = e[n], o = e[n + 1], s = this.calculateRangeRoute(i, o, t);
531
+ n === 0 && (i.distanceFromPrevious = 0, i.distanceFromStart = 0), o.distanceFromPrevious = this.calculateRouteDistance(s), o.distanceFromStart = l.roundPrecision((i.distanceFromStart || 0) + o.distanceFromPrevious);
532
532
  }
533
533
  return e;
534
534
  }
@@ -538,8 +538,8 @@ class S {
538
538
  * @param replace true replace the same waypoint with coordinate
539
539
  */
540
540
  static mergeCoordinatesToWaypoints(e, t, n = !0) {
541
- for (const o of e)
542
- this.mergeCoordinateToWaypoints(o, t, n);
541
+ for (const i of e)
542
+ this.mergeCoordinateToWaypoints(i, t, n);
543
543
  return t;
544
544
  }
545
545
  /**
@@ -556,12 +556,12 @@ class S {
556
556
  */
557
557
  static mergeCoordinateToWaypoints(e, t, n = !0) {
558
558
  e.lng = l.convertToStdLng(e.lng, 8);
559
- let o = Number.MAX_VALUE, i = 0, s = 0, r = 0;
559
+ let i = Number.MAX_VALUE, o = 0, s = 0, r = 0;
560
560
  for (let c = 0; c < t.length - 1; c++) {
561
561
  const a = { lng: t[c].lng, lat: t[c].lat }, d = { lng: t[c + 1].lng, lat: t[c + 1].lat }, u = this.calculatePointToLineDistance(e, a, d);
562
- o >= u && (o = u, i = c, s = this.calculateDistance(a, e, !1, 6), r = this.calculateDistance(d, e, !1, 6));
562
+ i >= u && (i = u, o = c, s = this.calculateDistance(a, e, !1, 6), r = this.calculateDistance(d, e, !1, 6));
563
563
  }
564
- return s !== 0 && r !== 0 ? s < o || s === o && i === 0 ? t.unshift(e) : r < o || r === o && i === t.length - 2 ? t.push(e) : t.splice(i + 1, 0, e) : s === 0 ? n && t.splice(i, 1, e) : r === 0 && n && t.splice(i + 1, 1, e), t.map((c) => (c.lng = l.convertToStdLng(c.lng), c));
564
+ return s !== 0 && r !== 0 ? s < i || s === i && o === 0 ? t.unshift(e) : r < i || r === i && o === t.length - 2 ? t.push(e) : t.splice(o + 1, 0, e) : s === 0 ? n && t.splice(o, 1, e) : r === 0 && n && t.splice(o + 1, 1, e), t.map((c) => (c.lng = l.convertToStdLng(c.lng), c));
565
565
  }
566
566
  /**
567
567
  * 生成航线(基于途经点生成大圆/横向航线,并根据是否跨180度分组)
@@ -571,12 +571,12 @@ class S {
571
571
  static generateRouteAccordingToWaypoints(e) {
572
572
  const t = [];
573
573
  for (let n = 1; n < e.length; n++) {
574
- const o = e[n - 1], i = e[n];
575
- if (n === 1 && t.push(o), i.gcToPrevious) {
576
- const s = this.interpolateCoordinates(o, i, 200, !1, !0, "nauticalmiles");
574
+ const i = e[n - 1], o = e[n];
575
+ if (n === 1 && t.push(i), o.gcToPrevious) {
576
+ const s = this.interpolateCoordinates(i, o, 200, !1, !0, "nauticalmiles");
577
577
  t.push(...s);
578
578
  } else
579
- t.push(i);
579
+ t.push(o);
580
580
  }
581
581
  return this.divideAccordingToLng(t, !0);
582
582
  }
@@ -586,12 +586,12 @@ class S {
586
586
  * @param route [[[lng, lat]]]
587
587
  */
588
588
  static nearestCoordinateInRoute(e, t) {
589
- const n = g.point([e.lng, e.lat]), o = [];
589
+ const n = g.point([e.lng, e.lat]), i = [];
590
590
  for (const c of t) {
591
591
  const a = c.map((d) => g.point(d));
592
- o.push(...a);
592
+ i.push(...a);
593
593
  }
594
- const i = g.featureCollection(o), s = g.nearestPoint(n, i), r = g.getCoord(s);
594
+ const o = g.featureCollection(i), s = g.nearestPoint(n, o), r = g.getCoord(s);
595
595
  return { lng: r[0], lat: r[1] };
596
596
  }
597
597
  /**
@@ -602,14 +602,14 @@ class S {
602
602
  static calculatePrevWaypoint(e, t) {
603
603
  let n = 0;
604
604
  this.mergeCoordinateToWaypoints(e, t);
605
- for (let o = 0; o < t.length - 1; o++) {
606
- const i = t[o], s = t[o + 1];
607
- if (this.calculateDistance(e, i) === 0) {
608
- n = o;
605
+ for (let i = 0; i < t.length - 1; i++) {
606
+ const o = t[i], s = t[i + 1];
607
+ if (this.calculateDistance(e, o) === 0) {
608
+ n = i;
609
609
  break;
610
610
  }
611
611
  if (this.calculateDistance(e, s) === 0) {
612
- n = o + 1;
612
+ n = i + 1;
613
613
  break;
614
614
  }
615
615
  }
@@ -623,40 +623,40 @@ class S {
623
623
  * @param units
624
624
  * @return { coordinate: {lng, lat}, route: [[[lng, lat]]]}
625
625
  */
626
- static calculateNextCoordinateAlongRoute(e, t, n, o = "nauticalmiles") {
626
+ static calculateNextCoordinateAlongRoute(e, t, n, i = "nauticalmiles") {
627
627
  var p;
628
- const i = e.speed || 12, s = [];
628
+ const o = e.speed || 12, s = [];
629
629
  let r = [], c = !1, a = 0, d = 0, u;
630
- if (t && n.length ? (s.push(e), n.forEach((f, M) => {
630
+ if (t && n.length ? (s.push(e), n.forEach((h, S) => {
631
631
  if (c)
632
- r.push(f);
632
+ r.push(h);
633
633
  else {
634
- const T = [];
635
- let P;
636
- for (let b = 0; b < f.length; b++)
634
+ const v = [];
635
+ let b;
636
+ for (let T = 0; T < h.length; T++)
637
637
  if (u)
638
- T.push(f[b]);
638
+ v.push(h[T]);
639
639
  else {
640
- P = { lng: f[b][0], lat: f[b][1] };
641
- const m = this.calculateDistance(e, P, !0, 8, o);
640
+ b = { lng: h[T][0], lat: h[T][1] };
641
+ const m = this.calculateDistance(e, b, !0, 8, i);
642
642
  if (a += m, a < t)
643
- d += m, s.push(P), e = P;
643
+ d += m, s.push(b), e = b;
644
644
  else {
645
645
  if (d = t, a === t)
646
- u = P, T.push([u.lng, u.lat]);
646
+ u = b, v.push([u.lng, u.lat]);
647
647
  else {
648
- const h = a - t, v = this.calculateBearing(P, e);
649
- u = this.calculateCoordinate(P, v, h, o), T.push([u.lng, u.lat]), T.push([P.lng, P.lat]);
648
+ const f = a - t, M = this.calculateBearing(b, e);
649
+ u = this.calculateCoordinate(b, M, f, i), v.push([u.lng, u.lat]), v.push([b.lng, b.lat]);
650
650
  }
651
651
  c = !0;
652
652
  }
653
653
  }
654
- T.length && r.push(T), M === n.length - 1 && !u && (u = P);
654
+ v.length && r.push(v), S === n.length - 1 && !u && (u = b);
655
655
  }
656
656
  })) : (r = n, u = { ...e }), u)
657
- if (s.push(u), u.distanceFromPrevious = d, u.hourFromPrevious = Math.round(d / i * 1e4) / 1e4, ((p = r[0]) == null ? void 0 : p.length) > 1) {
658
- const f = { lng: r[0][1][0], lat: r[0][1][1] };
659
- u.bearing = this.calculateBearing(u, f);
657
+ if (s.push(u), u.distanceFromPrevious = d, u.hourFromPrevious = Math.round(d / o * 1e4) / 1e4, ((p = r[0]) == null ? void 0 : p.length) > 1) {
658
+ const h = { lng: r[0][1][0], lat: r[0][1][1] };
659
+ u.bearing = this.calculateBearing(u, h);
660
660
  } else
661
661
  u.bearing = 0;
662
662
  return { coordinate: u, nextRoute: r, prevRoute: s };
@@ -668,10 +668,10 @@ class S {
668
668
  * @param to {lng, lat}
669
669
  */
670
670
  static nearestCoordinateInLine(e, t, n) {
671
- const o = l.convertToStdLng(e.lng, 6), i = g.point([o, e.lat]), s = l.convertToStdLng(t.lng, 6), r = l.convertToStdLng(n.lng, 6), c = g.lineString([
671
+ const i = l.convertToStdLng(e.lng, 6), o = g.point([i, e.lat]), s = l.convertToStdLng(t.lng, 6), r = l.convertToStdLng(n.lng, 6), c = g.lineString([
672
672
  [s, t.lat],
673
673
  [r, n.lat]
674
- ]), a = g.nearestPointOnLine(c, i), d = g.getCoord(a), u = l.roundPrecision(d[0], 6), p = l.roundPrecision(d[1], 6);
674
+ ]), a = g.nearestPointOnLine(c, o), d = g.getCoord(a), u = l.roundPrecision(d[0], 6), p = l.roundPrecision(d[1], 6);
675
675
  return { lng: u, lat: p, inline: !(u === s && p === t.lat) && !(u === r && p === n.lat) };
676
676
  }
677
677
  /**
@@ -681,21 +681,21 @@ class S {
681
681
  */
682
682
  static convertRouteToCoordinates(e, t = 0) {
683
683
  const n = [];
684
- let o, i;
684
+ let i, o;
685
685
  return e.forEach((s) => {
686
686
  s.forEach((r) => {
687
687
  const c = { lng: r[0], lat: r[1] };
688
- if (!i)
689
- n.push(c), i = c;
690
- else if (i.bearing === void 0)
691
- i.bearing = this.calculateBearing(i, c, !0);
688
+ if (!o)
689
+ n.push(c), o = c;
690
+ else if (o.bearing === void 0)
691
+ o.bearing = this.calculateBearing(o, c, !0);
692
692
  else {
693
- const a = this.calculateDistance(o, c, !0);
694
- a && a >= t && (o.bearing = this.calculateBearing(o, c, !0), n.push(o), i = o);
693
+ const a = this.calculateDistance(i, c, !0);
694
+ a && a >= t && (i.bearing = this.calculateBearing(i, c, !0), n.push(i), o = i);
695
695
  }
696
- o = c;
696
+ i = c;
697
697
  });
698
- }), o && n.push(o), n;
698
+ }), i && n.push(i), n;
699
699
  }
700
700
  /**
701
701
  * 抽稀(基于转向点)
@@ -704,8 +704,8 @@ class S {
704
704
  * @param distance
705
705
  */
706
706
  static simplifyRouteToCoordinates(e, t, n = 1) {
707
- let o = this.convertRouteToCoordinates(e, n);
708
- return o = this.simplifyGCCoordinates(o, t), o;
707
+ let i = this.convertRouteToCoordinates(e, n);
708
+ return i = this.simplifyGCCoordinates(i, t), i;
709
709
  }
710
710
  /**
711
711
  * 基于大圆标识抽稀
@@ -713,23 +713,23 @@ class S {
713
713
  * @param waypoints
714
714
  */
715
715
  static simplifyGCCoordinates(e, t) {
716
- t.forEach((o) => {
717
- this.mergeCoordinateToWaypoints(o, e);
716
+ t.forEach((i) => {
717
+ this.mergeCoordinateToWaypoints(i, e);
718
718
  });
719
- for (let o = 1; o < t.length; o++) {
720
- const i = t[o - 1], s = t[o];
719
+ for (let i = 1; i < t.length; i++) {
720
+ const o = t[i - 1], s = t[i];
721
721
  if (s.gcToPrevious) {
722
- const r = e.findIndex((a) => a.lng === i.lng && a.lat === i.lat), c = e.findIndex((a) => a.lng === s.lng && a.lat === s.lat);
722
+ const r = e.findIndex((a) => a.lng === o.lng && a.lat === o.lat), c = e.findIndex((a) => a.lng === s.lng && a.lat === s.lat);
723
723
  for (let a = c - 1; a > r; a--)
724
724
  e.splice(a, 1);
725
725
  }
726
726
  }
727
727
  let n = 0;
728
- for (let o = 1; o < e.length; o++) {
729
- const i = e[o - 1], s = e[o];
730
- s.gcToPrevious ? (i.bearing = this.calculateBearing(i, s, !1), s.distanceFromPrevious = this.calculateDistance(i, s, !1)) : (i.bearing = this.calculateBearing(i, s, !0), s.distanceFromPrevious = this.calculateDistance(i, s, !0)), n = l.roundPrecision(n + s.distanceFromPrevious), s.distanceFromStart = n;
728
+ for (let i = 1; i < e.length; i++) {
729
+ const o = e[i - 1], s = e[i];
730
+ s.gcToPrevious ? (o.bearing = this.calculateBearing(o, s, !1), s.distanceFromPrevious = this.calculateDistance(o, s, !1)) : (o.bearing = this.calculateBearing(o, s, !0), s.distanceFromPrevious = this.calculateDistance(o, s, !0)), n = l.roundPrecision(n + s.distanceFromPrevious), s.distanceFromStart = n;
731
731
  }
732
- return e.map((o) => (o.lng = l.convertToStdLng(o.lng), o));
732
+ return e.map((i) => (i.lng = l.convertToStdLng(i.lng), i));
733
733
  }
734
734
  /**
735
735
  * 计算轨迹中心点
@@ -740,8 +740,8 @@ class S {
740
740
  for (const r of e)
741
741
  for (const c of r)
742
742
  t.push(c);
743
- const n = g.featureCollection([]), o = l.convertToMonotonicLng2(t);
744
- for (const r of o)
743
+ const n = g.featureCollection([]), i = l.convertToMonotonicLng2(t);
744
+ for (const r of i)
745
745
  n.features.push(g.point(r));
746
746
  const s = g.center(n).geometry.coordinates;
747
747
  return { lng: l.convertToStdLng(s[0], 8), lat: l.roundPrecision(s[1], 8) };
@@ -760,11 +760,11 @@ class S {
760
760
  */
761
761
  static calculateBBox(e) {
762
762
  const t = [];
763
- for (const i of e)
764
- for (const s of i)
763
+ for (const o of e)
764
+ for (const s of o)
765
765
  t.push(s);
766
- const n = l.convertToMonotonicLng2(t), o = g.lineString(n);
767
- return g.bbox(o);
766
+ const n = l.convertToMonotonicLng2(t), i = g.lineString(n);
767
+ return g.bbox(i);
768
768
  }
769
769
  /**
770
770
  * 计算BBox
@@ -775,7 +775,7 @@ class S {
775
775
  return this.calculateBBox(t);
776
776
  }
777
777
  }
778
- const R = j.getLogger("vessel");
778
+ const N = F.getLogger("vessel");
779
779
  class q {
780
780
  /**
781
781
  * 将原始数据转换为geojson
@@ -784,20 +784,20 @@ class q {
784
784
  static convert2Geojson(e) {
785
785
  var n;
786
786
  const t = g.featureCollection([]);
787
- for (const o of e) {
788
- if (o.forecasts) {
789
- const i = (n = o.history) == null ? void 0 : n[0];
790
- for (const s of o.forecasts) {
791
- const r = [], c = N(s.date).utc(), a = `${o.name}-${s.model}`;
792
- if (i) {
793
- const d = N(i.updated).utc(), u = g.point([i.lng, i.lat], {
787
+ for (const i of e) {
788
+ if (i.forecasts) {
789
+ const o = (n = i.history) == null ? void 0 : n[0];
790
+ for (const s of i.forecasts) {
791
+ const r = [], c = D(s.date).utc(), a = `${i.name}-${s.model}`;
792
+ if (o) {
793
+ const d = D(o.updated).utc(), u = g.point([o.lng, o.lat], {
794
794
  model: s.model,
795
- name: o.name,
795
+ name: i.name,
796
796
  date: d.format(),
797
797
  hour: 0,
798
798
  format: d.format("MMM-DD/HHmm[Z]"),
799
- pressure: i.pressure > 1e4 ? l.roundPrecision(i.pressure / 100, 0) : l.roundPrecision(i.pressure, 0),
800
- wind: { kts: i.kts, spd: i.speed || i.spd },
799
+ pressure: o.pressure > 1e4 ? l.roundPrecision(o.pressure / 100, 0) : l.roundPrecision(o.pressure, 0),
800
+ wind: { kts: o.kts, spd: o.speed || o.spd },
801
801
  category: a,
802
802
  type: "forecast"
803
803
  });
@@ -806,9 +806,9 @@ class q {
806
806
  for (const d in s == null ? void 0 : s.hours) {
807
807
  const u = s.hours[d];
808
808
  u.wind.spd = u.wind.spd || u.wind.speed;
809
- const p = c.clone().add(Number(d), "hour"), f = g.point([u.lng, u.lat], {
809
+ const p = c.clone().add(Number(d), "hour"), h = g.point([u.lng, u.lat], {
810
810
  model: s.model,
811
- name: o.name,
811
+ name: i.name,
812
812
  date: p.format(),
813
813
  hour: Number(d),
814
814
  format: p.format("MMM-DD/HHmm[Z]"),
@@ -819,14 +819,14 @@ class q {
819
819
  category: a,
820
820
  type: "forecast"
821
821
  });
822
- t.features.push(f), r.push(f.geometry.coordinates);
822
+ t.features.push(h), r.push(h.geometry.coordinates);
823
823
  }
824
824
  if ((r == null ? void 0 : r.length) > 1) {
825
825
  const d = g.lineString(l.convertToMonotonicLng2(r), {
826
826
  date: s.date,
827
- id: o.id || o.name,
827
+ id: i.id || i.name,
828
828
  model: s.model,
829
- name: o.name,
829
+ name: i.name,
830
830
  category: a,
831
831
  type: "forecast"
832
832
  });
@@ -834,11 +834,11 @@ class q {
834
834
  }
835
835
  }
836
836
  }
837
- if (o.history) {
838
- const i = [];
839
- for (const r of o.history) {
840
- const c = N(r.updated).utc(), a = g.point([r.lng, r.lat], {
841
- name: o.name,
837
+ if (i.history) {
838
+ const o = [];
839
+ for (const r of i.history) {
840
+ const c = D(r.updated).utc(), a = g.point([r.lng, r.lat], {
841
+ name: i.name,
842
842
  date: c.format(),
843
843
  format: c.format("MMM-DD/HHmm[Z]"),
844
844
  pressure: r.pressure > 1e4 ? l.roundPrecision(r.pressure / 100, 0) : l.roundPrecision(r.pressure, 0),
@@ -847,14 +847,14 @@ class q {
847
847
  source: r.source,
848
848
  level: r.type,
849
849
  type: "history",
850
- category: `${o.name}-history`
850
+ category: `${i.name}-history`
851
851
  });
852
- t.features.push(a), i.push(a.geometry.coordinates);
852
+ t.features.push(a), o.push(a.geometry.coordinates);
853
853
  }
854
- const s = o.history[0];
855
- if (i.length === 1 && i.push(i[0]), i.length > 1) {
856
- const r = g.lineString(l.convertToMonotonicLng2(i), {
857
- name: o.name,
854
+ const s = i.history[0];
855
+ if (o.length === 1 && o.push(o[0]), o.length > 1) {
856
+ const r = g.lineString(l.convertToMonotonicLng2(o), {
857
+ name: i.name,
858
858
  type: "history",
859
859
  updated: s == null ? void 0 : s.updated,
860
860
  pressure: (s == null ? void 0 : s.pressure) > 1e4 ? l.roundPrecision((s == null ? void 0 : s.pressure) / 100, 0) : l.roundPrecision(s == null ? void 0 : s.pressure, 0),
@@ -875,68 +875,109 @@ class q {
875
875
  * @param step
876
876
  */
877
877
  static interpolate(e, t = 3) {
878
- var i, s, r, c;
879
- const n = (i = e == null ? void 0 : e.data) == null ? void 0 : i.features.filter((a) => a.geometry.type === "LineString" && a.properties.type === "forecast"), o = [];
878
+ var o, s, r, c;
879
+ const n = (o = e == null ? void 0 : e.data) == null ? void 0 : o.features.filter((a) => a.geometry.type === "LineString" && a.properties.type === "forecast"), i = [];
880
880
  for (const a of n) {
881
- const d = a.properties.name, u = a.properties.model, p = a.properties.showCircle, f = N(a.properties.date).utc();
882
- let M = t * 60 - (f.get("hour") * 60 + f.get("minute")) % (t * 60);
883
- const T = (s = e == null ? void 0 : e.data) == null ? void 0 : s.features.filter(
881
+ const d = a.properties.name, u = a.properties.model, p = a.properties.showCircle, h = D(a.properties.date).utc();
882
+ let S = t * 60 - (h.get("hour") * 60 + h.get("minute")) % (t * 60);
883
+ const v = (s = e == null ? void 0 : e.data) == null ? void 0 : s.features.filter(
884
884
  (m) => m.geometry.type === "Point" && m.properties.type === "forecast" && m.properties.category === `${d}-${u}`
885
885
  );
886
- let P, b = f.clone().add(M, "minute").set({ minute: 0, second: 0, millisecond: 0 });
887
- for (; P = this.pickIndex(T, b), P <= T.length - 1; ) {
888
- if (P > 0) {
889
- const m = T[P], h = P === 0 ? void 0 : T[P - 1], v = (M / 60 - ((r = h == null ? void 0 : h.properties) == null ? void 0 : r.hour)) / (m.properties.hour - ((c = h == null ? void 0 : h.properties) == null ? void 0 : c.hour)), y = this.computeNumber(h == null ? void 0 : h.geometry.coordinates[0], m.geometry.coordinates[0], v), I = this.computeNumber(h == null ? void 0 : h.geometry.coordinates[1], m.geometry.coordinates[1], v), D = g.point([y, I], {
886
+ let b, T = h.clone().add(S, "minute").set({ minute: 0, second: 0, millisecond: 0 });
887
+ for (; b = this.pickIndex(v, T), b <= v.length - 1; ) {
888
+ if (b > 0) {
889
+ const m = v[b], f = b === 0 ? void 0 : v[b - 1], M = (S / 60 - ((r = f == null ? void 0 : f.properties) == null ? void 0 : r.hour)) / (m.properties.hour - ((c = f == null ? void 0 : f.properties) == null ? void 0 : c.hour)), I = this.computeNumber(f == null ? void 0 : f.geometry.coordinates[0], m.geometry.coordinates[0], M), x = this.computeNumber(f == null ? void 0 : f.geometry.coordinates[1], m.geometry.coordinates[1], M), $ = g.point([I, x], {
890
890
  name: d,
891
891
  model: u,
892
892
  category: m == null ? void 0 : m.properties.category,
893
- date: b.format(),
894
- format: b.format("MMM-DD/HHmm[Z]"),
895
- gusts: this.computeNumber(h == null ? void 0 : h.properties.gusts, m.properties.gusts, v),
896
- hour: this.computeNumber(h == null ? void 0 : h.properties.hour, m.properties.hour, v),
897
- movement: this.computeNumber(h == null ? void 0 : h.properties.movement, m.properties.movement, v),
898
- pressure: this.computeNumber(h == null ? void 0 : h.properties.pressure, m.properties.pressure, v),
899
- wind: this.computeNumber(h == null ? void 0 : h.properties.wind, m.properties.wind, v),
893
+ date: T.format(),
894
+ format: T.format("MMM-DD/HHmm[Z]"),
895
+ gusts: this.computeNumber(f == null ? void 0 : f.properties.gusts, m.properties.gusts, M),
896
+ hour: this.computeNumber(f == null ? void 0 : f.properties.hour, m.properties.hour, M),
897
+ movement: this.computeNumber(f == null ? void 0 : f.properties.movement, m.properties.movement, M),
898
+ pressure: this.computeNumber(f == null ? void 0 : f.properties.pressure, m.properties.pressure, M),
899
+ wind: this.computeNumber(f == null ? void 0 : f.properties.wind, m.properties.wind, M),
900
900
  type: "forecast",
901
901
  showCircle: p
902
902
  });
903
- o.push(D);
903
+ i.push($);
904
904
  }
905
- M += t * 60, b = f.clone().add(M, "minute").set({ minute: 0, second: 0, millisecond: 0 });
905
+ S += t * 60, T = h.clone().add(S, "minute").set({ minute: 0, second: 0, millisecond: 0 });
906
906
  }
907
907
  }
908
- return o;
908
+ return i;
909
909
  }
910
910
  /**
911
- * 航行方向与台风
912
- * @param from
913
- * @param tropical
914
- * @param radius
915
- * @param speed
911
+ * 计算最佳绕航点
912
+ * 1) 确定当前船位与当前台风中心点之间的方位角,
913
+ * 2) 当前台风中心点与近24小时台风中心点的方位角,
914
+ * 3) 根据1)、2)及可航半圆计算绕航点
915
+ * @param from 船舶位置
916
+ * @param tropical 台风数据 { history: any[], forecasts: any[] }
917
+ * @param radius 与台风中心的距离
918
+ * @param speed 前进速度
919
+ * @param options
920
+ */
921
+ static diversionPassageAt(e, t, n, i, o = { requestId: "" }) {
922
+ var a, d, u, p;
923
+ const { t1: s, t2: r, hr: c } = this.tropicalCenterTwin(t, 24, o);
924
+ if (s && r) {
925
+ const h = P.calculateBearing(e, s), S = P.calculateBearing(s, r), v = P.calculateCoordinate(s, h - S + 90, n);
926
+ N.info("[%s] the right tangent position: %j", o.requestId, { from: e, t1: s, t2: r, radius: n, bearing1: h, bearing2: S, right: v });
927
+ const b = P.calculateBearing(e, v);
928
+ let T = P.calculateCoordinate(e, b, i * Number(c));
929
+ T.utc = D((d = (a = t == null ? void 0 : t.forecasts) == null ? void 0 : a[0]) == null ? void 0 : d.date).add(Number(c), "hours").utc().format(), N.info("[%s] the diversion position: %j", o.requestId, T);
930
+ const m = P.calculateDistance(T, r);
931
+ return N.info("[%s] the distance(%d) between the diversion position & t2: %j - %j", o.requestId, m, T, r), m < n && (T = P.calculateCoordinate(e, b, i * Number(c) / 2), T.utc = D((p = (u = t == null ? void 0 : t.forecasts) == null ? void 0 : u[0]) == null ? void 0 : p.date).add(Number(c) / 2, "hours").utc().format(), N.warn("[%s] the prev diversion position is too close to t2, replace with 1/2 position: %j", o.requestId, T)), T;
932
+ } else {
933
+ N.info("[%s] no need to diversion: %j", o.requestId, { from: e, t1: s, t2: r });
934
+ return;
935
+ }
936
+ }
937
+ /**
938
+ * 计算最佳漂航点
939
+ * 1) 确定当前船位与当前台风中心点之间的方位角,
940
+ * 2) 当前台风中心点与近24小时台风中心点的方位角,
941
+ * 3) 根据2)、3)反向计算航漂航点
942
+ * @param from 船舶位置
943
+ * @param tropical 台风数据 { history: any[], forecasts: any[] }
944
+ * @param radius 与台风中心的距离
916
945
  * @param options
917
946
  */
918
- static diversionPassageAt(e, t, n, o, i = { requestId: "" }) {
919
- var d, u, p, f, M, T, P, b, m;
920
- const s = (u = (d = t == null ? void 0 : t.forecasts) == null ? void 0 : d[0]) == null ? void 0 : u.hours, r = ((p = t == null ? void 0 : t.history) == null ? void 0 : p[0]) || (s == null ? void 0 : s[(f = Object.keys(s)) == null ? void 0 : f[0]]);
921
- R.info("[%s] the first evasion point: %j", i.requestId, r);
922
- const c = (M = Object.keys(s || {}).filter((h) => Number(h) <= 24)) == null ? void 0 : M.at(-1), a = s == null ? void 0 : s[c || -1];
923
- if (R.info("[%s] the second evasion point in %s hrs: %j", i.requestId, c, r), r && a) {
924
- const h = S.calculateBearing(e, r), v = S.calculateBearing(r, a), y = S.calculateCoordinate(r, h - v + 90, n);
925
- R.info("[%s] the right tangent point: %j", i.requestId, { from: e, p1: r, p2: a, radius: n, bearing1: h, bearing2: v, right: y });
926
- const I = S.calculateBearing(e, y);
927
- let D = S.calculateCoordinate(e, I, o * Number(c));
928
- D.utc = N((P = (T = t == null ? void 0 : t.forecasts) == null ? void 0 : T[0]) == null ? void 0 : P.date).add(Number(c), "hours").utc().format(), R.info("[%s] the temp evasion point: %j", i.requestId, D);
929
- const $ = S.calculateDistance(D, a);
930
- return R.info("[%s] the distance(%d) between tmp & p2: %j - %j", i.requestId, $, D, a), $ < n && (D = S.calculateCoordinate(e, I, o * Number(c) / 2), D.utc = N((m = (b = t == null ? void 0 : t.forecasts) == null ? void 0 : b[0]) == null ? void 0 : m.date).add(Number(c) / 2, "hours").utc().format(), R.warn("[%s] tmp is too close to p2, replace with 1/2 position: %j", i.requestId, D)), D;
947
+ static driftPassageAt(e, t, n, i = { requestId: "" }) {
948
+ const { t1: o, t2: s, hr: r } = this.tropicalCenterTwin(t, 24, i);
949
+ if (o && s) {
950
+ const c = P.calculateBearing(e, o), a = P.calculateBearing(o, s);
951
+ return P.calculateCoordinate(o, c - a + 180, n);
931
952
  } else {
932
- R.info("[%s] no need evasion: %j", i.requestId, { from: e, p1: r, p2: a });
953
+ N.info("[%s] no need drift: %j", i.requestId, { from: e, t1: o, t2: s });
933
954
  return;
934
955
  }
935
956
  }
957
+ /**
958
+ * 获取台风中心点对
959
+ * @param tropical { history: any[], forecasts: any[] }
960
+ * @param hrs 未来小时数,默认24
961
+ * @param options
962
+ * @private
963
+ */
964
+ static tropicalCenterTwin(e, t = 24, n = { requestId: "" }) {
965
+ var c, a, d, u, p;
966
+ let i = {};
967
+ (c = e.forecasts) == null || c.forEach((h) => {
968
+ i = { ...h.hours, ...i };
969
+ });
970
+ const o = ((a = e == null ? void 0 : e.history) == null ? void 0 : a[0]) || (i == null ? void 0 : i[(d = Object.keys(i)) == null ? void 0 : d[0]]);
971
+ N.info("[%s] the first tropical center: %j", n.requestId, o);
972
+ let s = (u = Object.keys(i || {}).filter((h) => Number(h) <= (t < 0 ? 24 : t))) == null ? void 0 : u.at(-1);
973
+ s || (s = (p = Object.keys(i || {}).filter((h) => Number(h) <= (t < 0 ? 24 : 2 * t))) == null ? void 0 : p.at(-1));
974
+ const r = i == null ? void 0 : i[s || -1];
975
+ return N.info("[%s] the second tropical center: %j in %d hrs", n.requestId, r, s), { t1: o, t2: r, hr: s };
976
+ }
936
977
  static pickIndex(e, t) {
937
978
  let n = 0;
938
- for (const o of e) {
939
- if (N(o.properties.date).isAfter(t))
979
+ for (const i of e) {
980
+ if (D(i.properties.date).isAfter(t))
940
981
  return n === 0 ? -1 : n;
941
982
  n++;
942
983
  }
@@ -946,10 +987,10 @@ class q {
946
987
  if (e)
947
988
  if (t) {
948
989
  if (isNaN(e) && isNaN(t) && typeof e != "string" && typeof t != "string") {
949
- const o = {};
950
- for (const i in e)
951
- o[i] = this.computeNumber(e[i], t[i], n);
952
- return o;
990
+ const i = {};
991
+ for (const o in e)
992
+ i[o] = this.computeNumber(e[o], t[o], n);
993
+ return i;
953
994
  }
954
995
  return Math.round((e + (t - e) * n) * 100) / 100;
955
996
  } else
@@ -959,7 +1000,7 @@ class q {
959
1000
  }
960
1001
  }
961
1002
  export {
962
- S as LaneHelper,
1003
+ P as LaneHelper,
963
1004
  l as LngLatHelper,
964
1005
  q as TropicalHelper
965
1006
  };
@@ -1 +1 @@
1
- (function(C,I){typeof exports=="object"&&typeof module<"u"?I(exports,require("@turf/turf"),require("moment"),require("moment-timezone"),require("tz-lookup")):typeof define=="function"&&define.amd?define(["exports","@turf/turf","moment","moment-timezone","tz-lookup"],I):(C=typeof globalThis<"u"?globalThis:C||self,I(C["idm-plugin-rabbitmq"]={},C["@turf/turf"],C.moment,C["moment-timezone"],C["tz-lookup"]))})(this,function(C,I,y,q,W){"use strict";function F(T){const e=Object.create(null,{[Symbol.toStringTag]:{value:"Module"}});if(T){for(const t in T)if(t!=="default"){const n=Object.getOwnPropertyDescriptor(T,t);Object.defineProperty(e,t,n.get?n:{enumerable:!0,get:()=>T[t]})}}return e.default=T,Object.freeze(e)}const g=F(I);function O(T){return T&&T.__esModule&&Object.prototype.hasOwnProperty.call(T,"default")?T.default:T}class j{log(){}isLevelEnabled(){return!1}addContext(){}removeContext(){}clearContext(){}}["Trace","Debug","Info","Warn","Error","Fatal","Mark"].forEach(T=>{j.prototype[T.toLowerCase()]=()=>{},j.prototype[`is${T}Enabled`]=()=>!1});const k=(()=>{try{return require("log4js")}catch{return null}})();var E={getLogger:k?k.getLogger:()=>new j};const A=O(E);class l{static guessTimeZoneOffset(e,t){const n=W(t,e),o=y().tz(n).utcOffset();return this.roundPrecision(o/60,1)}static prettyTimeZoneOffset(e){let t=Math.floor(Math.abs(e)),n=Math.round((Math.abs(e)-t)*60);return n=n>9?n:`0${n}`,t=t>9?t:`0${t}`,e>0?`+${t}:${n}`:`-${t}:${n}`}static lng2pretty(e,t=6,n="H°M′"){e=l.convertToStdLng(e,t);let o="E";e<0&&(o="W"),e=Math.abs(e),n=n.toUpperCase();let i=e*3600,s,r,c,u,d,a;s=i%3600%60,n.indexOf("S")!==-1&&(i=i-s,r=l.padNumber(s,2,2)),c=i/60%60,n.indexOf("M")!==-1&&(n.indexOf("S")!==-1?u=l.roundPrecision(c,t).toString().padStart(2,"0"):u=l.padNumber(c,2,2),i=i-c*60),d=i/3600,n.indexOf("M")!==-1?a=l.roundPrecision(d,t).toString().padStart(3,"0"):a=l.padNumber(d,3,2);const p=`${n.replace(/S+/gi,r).replace(/M+/gi,u).replace(/H+/gi,a)}${o}`;return{direction:o,degree:l.roundPrecision(d,t),minute:l.roundPrecision(c,t),second:l.roundPrecision(s,t),pretty:p}}static lat2pretty(e,t=6,n="H°M′"){e=e%180;let o="N";e<0&&(o="S"),e=Math.abs(e),n=n.toUpperCase();let i=e*3600,s,r,c,u,d,a;s=i%3600%60,n.indexOf("S")!==-1&&(i=i-s,r=l.padNumber(s,2,2)),c=i/60%60,n.indexOf("M")!==-1&&(n.indexOf("S")!==-1?u=l.roundPrecision(c,t).toString().padStart(2,"0"):u=l.padNumber(c,2,2),i=i-c*60),d=i/3600,n.indexOf("M")!==-1?a=l.roundPrecision(d,t).toString().padStart(2,"0"):a=l.padNumber(d,2,2);const p=`${n.replace(/S+/gi,r).replace(/M+/gi,u).replace(/H+/gi,a)}${o}`;return{direction:o,degree:l.roundPrecision(d,t),minute:l.roundPrecision(c,t),second:l.roundPrecision(s,t),pretty:p}}static str2Lng(e,t=6){let n;if(isNaN(e)){e=l.strReplace(e,"LNG");const o=e[e.length-1].toUpperCase();e=e.substring(0,e.length-1).trim();const i=e.split(" ").filter(c=>c!=="").map(c=>Number(c));let[s,r]=i;if(r=r>60?r/Math.pow(10,String(r).length-2):r,s>360&&!r){const c=this.roundPrecision(s/100,0);r=s-c*100,s=c}n=s+(r??0)/60,o==="W"&&(n=n*-1)}else n=Number(e);return l.convertToStdLng(n,t)}static str2Lat(e,t=6){let n;if(isNaN(e)){e=l.strReplace(e,"LAT");const o=e[e.length-1].toUpperCase();e=e.substring(0,e.length-1).trim();const i=e.split(" ").filter(c=>c!=="").map(c=>Number(c));let[s,r]=i;if(r=r>60?r/Math.pow(10,String(r).length-2):r,s>90&&!r){const c=this.roundPrecision(s/100,0);r=s-c*100,s=c}n=s+(r??0)/60,o==="S"&&(n=n*-1)}else n=Number(e);return l.roundPrecision(n,t)}static str2LngOrLat(e,t=6,n="LAT"){e=l.strReplace(e,n);const o=e[e.length-1].toUpperCase();return["N","S"].includes(o)?{lat:l.str2Lat(e,t)}:{lng:l.str2Lng(e,t)}}static convertToStdLng(e,t=4){return e>180?(e=e%360,e=e>180?e-360:e):e<-180&&(e=e%360,e=e<-180?e+360:e),l.roundPrecision(e,t)}static roundPrecision(e,t=4){if(typeof e=="number"){const n=Number("1".padEnd(t+1,"0"));return Math.round(e*n)/n}return e}static convertToMonotonicLng2(e){for(let t=1;t<e.length;t++)e[t][0]+=Math.round((e[t-1][0]-e[t][0])/360)*360;return e}static convertToMonotonicLng(e){for(let t=1;t<e.length;t++)e[t].lng+=Math.round((e[t-1].lng-e[t].lng)/360)*360;return e}static strReplace(e,t="LAT"){e=e.replace(/([0-9]+)\.([0-9]+\.[0-9]+)/g,"$1 $2").replace(/-/g," ").replace(/°/," ").replace(/'/g," ").replace(/′/g," ").replace(/"/g," ").replace(/∼/g," ").replace(/°/g," ").replace(/,/g,".").replace(/^ /g,"").replace(/ $/g,"").trim();const n=e[e.length-1].toUpperCase();if(!["N","S","E","W"].includes(n)){const o=e,i=Number(o.split(" ")[0]);if(isNaN(i))throw new Error(`invalid Lat/Lng: ${e}`);i>=90?e=`${o}E`:i<=-90?e=`${o}W`:["LAN","LNG"].includes(t==null?void 0:t.toUpperCase())?e=`${o}${i>0?"E":"W"}`:e=`${o}${i>0?"N":"S"}`}return e}static padNumber(e,t=2,n=2){const o=Math.trunc(e).toString().padStart(t,"0"),i=Math.trunc(l.roundPrecision(e-Math.trunc(e),n)*Math.pow(10,n)).toString().padStart(n,"0");return`${o}.${i}`}}class v{static calculateBearing(e,t,n=!0,o=4){const i=g.points([[e.lng,e.lat],[t.lng,t.lat]]);let s;return n?s=g.rhumbBearing(i.features[0],i.features[1]):s=g.bearing(i.features[0],i.features[1]),s<0&&(s+=360),l.roundPrecision(s,o)}static calculateDistance(e,t,n=!0,o=4,i="nauticalmiles"){e={...e},t={...t},e.lng=l.convertToStdLng(e.lng,o),t.lng=l.convertToStdLng(t.lng,o);const s=g.points([[e.lng,e.lat],[t.lng,t.lat]]);let r;return n?r=g.rhumbDistance(s.features[0],s.features[1],{units:i}):r=g.distance(s.features[0],s.features[1],{units:i}),l.roundPrecision(r,o)}static calculateRouteDistance(e,t=4,n="nauticalmiles"){let o=0,i;for(const s of e)for(let r=0;r<s.length-1;r++){const c={lng:s[r][0],lat:s[r][1]};r===0&&i&&(o+=this.calculateDistance(i,c,!0,t,n));const u={lng:s[r+1][0],lat:s[r+1][1]};o+=this.calculateDistance(c,u,!0,t,n),i=u}return l.roundPrecision(o,t)}static calculateCoordinate(e,t,n,o="nauticalmiles",i=!0){const s=g.point([e.lng,e.lat]);let r;i?r=g.rhumbDestination(s,n,t,{units:o}):r=g.destination(s,n,t,{units:o});const c=r.geometry.coordinates;return{lng:l.convertToStdLng(c[0],8),lat:l.roundPrecision(c[1],8)}}static interpolateCoordinates(e,t,n,o=!0,i=!0,s="nauticalmiles"){const r=[],c=this.calculateBearing(e,t,!1),u=this.calculateDistance(e,t,!1,8,s);o&&r.push({lng:e.lng,lat:e.lat});let d=0;for(;d<u;)d+=n,d<u&&r.push(this.calculateCoordinate(e,c,d,s,!1));return i&&r.push({lng:t.lng,lat:t.lat}),r}static divideAccordingToLng(e,t=!1){if((e==null?void 0:e.length)<2)return[];e=this.deduplicateCoordinates(e);let n=[];const o=[];let i,s;for(let r=0;r<e.length-1;r++){i=l.convertToStdLng(e[r].lng,8),s=l.convertToStdLng(e[r+1].lng,8),n.push([i,e[r].lat]);const c=i-s;if(Math.abs(c)>180){const u=l.convertToMonotonicLng2([[i,e[r].lat],[s,e[r+1].lat]]);let d,a;t?(d=g.lineString(u),a=g.lineString([[c>0?180:-180,89],[c>0?180:-180,-89]])):(d=g.greatCircle(u[0],u[1]),a=g.greatCircle([c>0?180:-180,89],[c>0?180:-180,-89]));const p=g.lineIntersect(d,a);let h;if(p.features.length){const D=g.getCoord(p.features[0]);h=l.roundPrecision(D[1],8)}else h=e[r].lat;c>0?(n.push([180-1e-6,h]),o.push([...n]),n=[],n.push([-(180-1e-6),h])):(n.push([-(180-1e-6),h]),o.push([...n]),n=[],n.push([180-1e-6,h]))}r===e.length-2&&n.push([s,e[r+1].lat])}return o.push(n),o}static deduplicateRoute(e){const t=[];for(const n of e){const o=n.reduce((i,s)=>(i.findIndex(r=>r[0]===s[0]&&r[1]===s[1])===-1&&i.push(s),i),[]);t.push(o)}return t}static deduplicateCoordinates(e){return e.reduce((t,n)=>(t.findIndex(o=>o.lat===n.lat&&o.lng===n.lng)===-1&&t.push(n),t),[])}static removeCoordinateFromRoute(e,t){e.lng=l.convertToStdLng(e.lng,8);for(const n of t)for(let o=n.length-1;o>=0;o--)l.roundPrecision(n[o][0],8)===e.lng&&l.roundPrecision(n[o][1],8)===l.roundPrecision(e.lat,8)&&n.splice(o,1);return t}static removeCoordinateFromWaypoints(e,t){e.lng=l.convertToStdLng(e.lng,8);for(let n=t.length-1;n>=0;n--)l.roundPrecision(t[n].lng,8)===e.lng&&l.roundPrecision(t[n].lat,8)===l.roundPrecision(e.lat,8)&&t.splice(n,1);return t}static mergeCoordinateToRoute(e,t){e.lng=l.convertToStdLng(e.lng,8);let n=Number.MAX_VALUE,o=0,i=0,s,r;return t.forEach((c,u)=>{for(let d=0;d<c.length-1;d++){const a={lng:c[d][0],lat:c[d][1]},p={lng:c[d+1][0],lat:c[d+1][1]},h=this.calculatePointToLineDistance(e,a,p);n>h&&(n=h,i=d,o=u,s=this.calculateDistance(a,e),r=this.calculateDistance(p,e))}}),s!==0&&r!==0?t[o].splice(i+1,0,[e.lng,e.lat]):s===0?t[o].splice(i,1,[e.lng,e.lat]):r===0&&t[o].splice(i+1,1,[e.lng,e.lat]),t}static appendCoordinateToRoute(e,t){e.lng=l.convertToStdLng(e.lng,8);const n=v.convertRouteToCoordinates(t);return n.push(e),v.divideAccordingToLng(n)}static unshiftCoordinateToRoute(e,t){const n=v.convertRouteToCoordinates(t);return n.unshift(e),v.divideAccordingToLng(n)}static mergeWaypointsToRoute(e,t){for(const n of e)t=this.mergeCoordinateToRoute(n,t);return t}static calculateRangeRoute(e,t,n){n=this.mergeWaypointsToRoute([e,t],n);const o=[];let i=0;return n.forEach(s=>{if(i===2)return;const r=[];for(const c of s){if(l.roundPrecision(t.lng,8)===l.roundPrecision(c[0],8)&&l.roundPrecision(t.lat,8)===l.roundPrecision(c[1],8)){r.push(c),i===0&&r.push([e.lng,e.lat]),i=2;break}i===1?r.push(c):l.roundPrecision(e.lng,8)===l.roundPrecision(c[0],8)&&l.roundPrecision(e.lat,8)===l.roundPrecision(c[1],8)&&(i=1,r.push(c))}r.length&&o.push(r)}),o}static calculateRangeWaypoints(e,t,n,o=[]){const i=this.convertRouteToCoordinates(n,0),s=this.mergeCoordinatesToWaypoints([e,t,...o],i),r=s.findIndex(d=>l.roundPrecision(e.lng,8)===l.roundPrecision(d.lng,8)&&l.roundPrecision(e.lat,8)===l.roundPrecision(d.lat,8)),c=s.findIndex(d=>l.roundPrecision(t.lng,8)===l.roundPrecision(d.lng,8)&&l.roundPrecision(t.lat,8)===l.roundPrecision(d.lat,8));return s.filter((d,a)=>a>=r&&a<=c)}static calculateMinDistanceToRoute(e,t){let n=Number.MAX_VALUE,o=0,i=0;return t.forEach((s,r)=>{for(let c=0;c<s.length-1;c++){const u={lng:s[c][0],lat:s[c][1]},d={lng:s[c+1][0],lat:s[c+1][1]},a=this.calculatePointToLineDistance(e,u,d);n>a&&(n=a,o=c,i=r)}}),{minDist:n,segIndex:i,minIndex:o}}static calculateSubRoute(e,t){const{segIndex:n,minIndex:o}=this.calculateMinDistanceToRoute({...e},t);e.lng=l.convertToStdLng(e.lng);const i=[];let s=!0;for(let r=n;r<t.length;r++)if(s){const c=[];c.push([e.lng,e.lat]);for(let u=o+1;u<t[r].length;u++)e.lng===t[r][u][0]&&e.lat===t[r][u][1]||c.push(t[r][u]);i.push(c),s=!1}else i.push([...t[r]]);return i}static calculateSubWaypoints(e,t){let n=Number.MAX_VALUE,o=0;for(let s=0;s<t.length-1;s++){const r=t[s],c=t[s+1];if(this.calculateDistance(e,r)===0)return t;if(this.calculateDistance(e,c)===0)return t.filter((d,a)=>a>0);const u=this.calculatePointToLineDistance(e,r,c);n>u&&(n=u,o=s)}e.lng=l.convertToStdLng(e.lng);const i=[e];for(let s=o+1;s<t.length;s++)i.push(t[s]);return i}static calculatePointToLineDistance(e,t,n,o={units:"nauticalmiles",method:"geodesic"}){e.lng=l.convertToStdLng(e.lng),t={...t},n={...n},t.lng=l.convertToStdLng(t.lng,8),n.lng=l.convertToStdLng(n.lng,8);const i=l.convertToMonotonicLng([t,n]);t=i[0],n=i[1];const s=g.lineString([[t.lng,t.lat],[n.lng,n.lat]]),r=g.pointToLineDistance(g.point([e.lng,e.lat]),s,o),c=g.pointToLineDistance(g.point([e.lng>0?e.lng-360:e.lng+360,e.lat]),s,o);return l.roundPrecision(Math.min(r,c),6)}static calculateWaypointsPropInRoute(e,t){t=this.mergeWaypointsToRoute(e,t);for(let n=0;n<e.length-1;n++){const o=e[n],i=e[n+1],s=this.calculateRangeRoute(o,i,t);n===0&&(o.distanceFromPrevious=0,o.distanceFromStart=0),i.distanceFromPrevious=this.calculateRouteDistance(s),i.distanceFromStart=l.roundPrecision((o.distanceFromStart||0)+i.distanceFromPrevious)}return e}static mergeCoordinatesToWaypoints(e,t,n=!0){for(const o of e)this.mergeCoordinateToWaypoints(o,t,n);return t}static mergeCoordinateToWaypoints(e,t,n=!0){e.lng=l.convertToStdLng(e.lng,8);let o=Number.MAX_VALUE,i=0,s=0,r=0;for(let c=0;c<t.length-1;c++){const u={lng:t[c].lng,lat:t[c].lat},d={lng:t[c+1].lng,lat:t[c+1].lat},a=this.calculatePointToLineDistance(e,u,d);o>=a&&(o=a,i=c,s=this.calculateDistance(u,e,!1,6),r=this.calculateDistance(d,e,!1,6))}return s!==0&&r!==0?s<o||s===o&&i===0?t.unshift(e):r<o||r===o&&i===t.length-2?t.push(e):t.splice(i+1,0,e):s===0?n&&t.splice(i,1,e):r===0&&n&&t.splice(i+1,1,e),t.map(c=>(c.lng=l.convertToStdLng(c.lng),c))}static generateRouteAccordingToWaypoints(e){const t=[];for(let n=1;n<e.length;n++){const o=e[n-1],i=e[n];if(n===1&&t.push(o),i.gcToPrevious){const s=this.interpolateCoordinates(o,i,200,!1,!0,"nauticalmiles");t.push(...s)}else t.push(i)}return this.divideAccordingToLng(t,!0)}static nearestCoordinateInRoute(e,t){const n=g.point([e.lng,e.lat]),o=[];for(const c of t){const u=c.map(d=>g.point(d));o.push(...u)}const i=g.featureCollection(o),s=g.nearestPoint(n,i),r=g.getCoord(s);return{lng:r[0],lat:r[1]}}static calculatePrevWaypoint(e,t){let n=0;this.mergeCoordinateToWaypoints(e,t);for(let o=0;o<t.length-1;o++){const i=t[o],s=t[o+1];if(this.calculateDistance(e,i)===0){n=o;break}if(this.calculateDistance(e,s)===0){n=o+1;break}}return t[n===0?0:n-1]}static calculateNextCoordinateAlongRoute(e,t,n,o="nauticalmiles"){var p;const i=e.speed||12,s=[];let r=[],c=!1,u=0,d=0,a;if(t&&n.length?(s.push(e),n.forEach((h,D)=>{if(c)r.push(h);else{const P=[];let b;for(let S=0;S<h.length;S++)if(a)P.push(h[S]);else{b={lng:h[S][0],lat:h[S][1]};const m=this.calculateDistance(e,b,!0,8,o);if(u+=m,u<t)d+=m,s.push(b),e=b;else{if(d=t,u===t)a=b,P.push([a.lng,a.lat]);else{const f=u-t,M=this.calculateBearing(b,e);a=this.calculateCoordinate(b,M,f,o),P.push([a.lng,a.lat]),P.push([b.lng,b.lat])}c=!0}}P.length&&r.push(P),D===n.length-1&&!a&&(a=b)}})):(r=n,a={...e}),a)if(s.push(a),a.distanceFromPrevious=d,a.hourFromPrevious=Math.round(d/i*1e4)/1e4,((p=r[0])==null?void 0:p.length)>1){const h={lng:r[0][1][0],lat:r[0][1][1]};a.bearing=this.calculateBearing(a,h)}else a.bearing=0;return{coordinate:a,nextRoute:r,prevRoute:s}}static nearestCoordinateInLine(e,t,n){const o=l.convertToStdLng(e.lng,6),i=g.point([o,e.lat]),s=l.convertToStdLng(t.lng,6),r=l.convertToStdLng(n.lng,6),c=g.lineString([[s,t.lat],[r,n.lat]]),u=g.nearestPointOnLine(c,i),d=g.getCoord(u),a=l.roundPrecision(d[0],6),p=l.roundPrecision(d[1],6);return{lng:a,lat:p,inline:!(a===s&&p===t.lat)&&!(a===r&&p===n.lat)}}static convertRouteToCoordinates(e,t=0){const n=[];let o,i;return e.forEach(s=>{s.forEach(r=>{const c={lng:r[0],lat:r[1]};if(!i)n.push(c),i=c;else if(i.bearing===void 0)i.bearing=this.calculateBearing(i,c,!0);else{const u=this.calculateDistance(o,c,!0);u&&u>=t&&(o.bearing=this.calculateBearing(o,c,!0),n.push(o),i=o)}o=c})}),o&&n.push(o),n}static simplifyRouteToCoordinates(e,t,n=1){let o=this.convertRouteToCoordinates(e,n);return o=this.simplifyGCCoordinates(o,t),o}static simplifyGCCoordinates(e,t){t.forEach(o=>{this.mergeCoordinateToWaypoints(o,e)});for(let o=1;o<t.length;o++){const i=t[o-1],s=t[o];if(s.gcToPrevious){const r=e.findIndex(u=>u.lng===i.lng&&u.lat===i.lat),c=e.findIndex(u=>u.lng===s.lng&&u.lat===s.lat);for(let u=c-1;u>r;u--)e.splice(u,1)}}let n=0;for(let o=1;o<e.length;o++){const i=e[o-1],s=e[o];s.gcToPrevious?(i.bearing=this.calculateBearing(i,s,!1),s.distanceFromPrevious=this.calculateDistance(i,s,!1)):(i.bearing=this.calculateBearing(i,s,!0),s.distanceFromPrevious=this.calculateDistance(i,s,!0)),n=l.roundPrecision(n+s.distanceFromPrevious),s.distanceFromStart=n}return e.map(o=>(o.lng=l.convertToStdLng(o.lng),o))}static calculateCenter(e){const t=[];for(const r of e)for(const c of r)t.push(c);const n=g.featureCollection([]),o=l.convertToMonotonicLng2(t);for(const r of o)n.features.push(g.point(r));const s=g.center(n).geometry.coordinates;return{lng:l.convertToStdLng(s[0],8),lat:l.roundPrecision(s[1],8)}}static calculateCenter2(e){const t=this.generateRouteAccordingToWaypoints(e);return this.calculateCenter(t)}static calculateBBox(e){const t=[];for(const i of e)for(const s of i)t.push(s);const n=l.convertToMonotonicLng2(t),o=g.lineString(n);return g.bbox(o)}static calculateBBox2(e){const t=this.generateRouteAccordingToWaypoints(e);return this.calculateBBox(t)}}const R=A.getLogger("vessel");class B{static convert2Geojson(e){var n;const t=g.featureCollection([]);for(const o of e){if(o.forecasts){const i=(n=o.history)==null?void 0:n[0];for(const s of o.forecasts){const r=[],c=y(s.date).utc(),u=`${o.name}-${s.model}`;if(i){const d=y(i.updated).utc(),a=g.point([i.lng,i.lat],{model:s.model,name:o.name,date:d.format(),hour:0,format:d.format("MMM-DD/HHmm[Z]"),pressure:i.pressure>1e4?l.roundPrecision(i.pressure/100,0):l.roundPrecision(i.pressure,0),wind:{kts:i.kts,spd:i.speed||i.spd},category:u,type:"forecast"});t.features.push(a),r.push(a.geometry.coordinates)}for(const d in s==null?void 0:s.hours){const a=s.hours[d];a.wind.spd=a.wind.spd||a.wind.speed;const p=c.clone().add(Number(d),"hour"),h=g.point([a.lng,a.lat],{model:s.model,name:o.name,date:p.format(),hour:Number(d),format:p.format("MMM-DD/HHmm[Z]"),pressure:a.pressure>1e4?l.roundPrecision(a.pressure/100,0):l.roundPrecision(a.pressure,0),gusts:a.gusts,wind:a.wind||{},movement:a.movement,category:u,type:"forecast"});t.features.push(h),r.push(h.geometry.coordinates)}if((r==null?void 0:r.length)>1){const d=g.lineString(l.convertToMonotonicLng2(r),{date:s.date,id:o.id||o.name,model:s.model,name:o.name,category:u,type:"forecast"});t.features.push(d)}}}if(o.history){const i=[];for(const r of o.history){const c=y(r.updated).utc(),u=g.point([r.lng,r.lat],{name:o.name,date:c.format(),format:c.format("MMM-DD/HHmm[Z]"),pressure:r.pressure>1e4?l.roundPrecision(r.pressure/100,0):l.roundPrecision(r.pressure,0),spd:r.speed||r.spd,kts:r.kts,source:r.source,level:r.type,type:"history",category:`${o.name}-history`});t.features.push(u),i.push(u.geometry.coordinates)}const s=o.history[0];if(i.length===1&&i.push(i[0]),i.length>1){const r=g.lineString(l.convertToMonotonicLng2(i),{name:o.name,type:"history",updated:s==null?void 0:s.updated,pressure:(s==null?void 0:s.pressure)>1e4?l.roundPrecision((s==null?void 0:s.pressure)/100,0):l.roundPrecision(s==null?void 0:s.pressure,0),spd:(s==null?void 0:s.speed)||(s==null?void 0:s.spd),kts:s==null?void 0:s.kts,source:s==null?void 0:s.source,level:s==null?void 0:s.type});t.features.push(r)}}}return t}static interpolate(e,t=3){var i,s,r,c;const n=(i=e==null?void 0:e.data)==null?void 0:i.features.filter(u=>u.geometry.type==="LineString"&&u.properties.type==="forecast"),o=[];for(const u of n){const d=u.properties.name,a=u.properties.model,p=u.properties.showCircle,h=y(u.properties.date).utc();let D=t*60-(h.get("hour")*60+h.get("minute"))%(t*60);const P=(s=e==null?void 0:e.data)==null?void 0:s.features.filter(m=>m.geometry.type==="Point"&&m.properties.type==="forecast"&&m.properties.category===`${d}-${a}`);let b,S=h.clone().add(D,"minute").set({minute:0,second:0,millisecond:0});for(;b=this.pickIndex(P,S),b<=P.length-1;){if(b>0){const m=P[b],f=b===0?void 0:P[b-1],M=(D/60-((r=f==null?void 0:f.properties)==null?void 0:r.hour))/(m.properties.hour-((c=f==null?void 0:f.properties)==null?void 0:c.hour)),x=this.computeNumber(f==null?void 0:f.geometry.coordinates[0],m.geometry.coordinates[0],M),$=this.computeNumber(f==null?void 0:f.geometry.coordinates[1],m.geometry.coordinates[1],M),N=g.point([x,$],{name:d,model:a,category:m==null?void 0:m.properties.category,date:S.format(),format:S.format("MMM-DD/HHmm[Z]"),gusts:this.computeNumber(f==null?void 0:f.properties.gusts,m.properties.gusts,M),hour:this.computeNumber(f==null?void 0:f.properties.hour,m.properties.hour,M),movement:this.computeNumber(f==null?void 0:f.properties.movement,m.properties.movement,M),pressure:this.computeNumber(f==null?void 0:f.properties.pressure,m.properties.pressure,M),wind:this.computeNumber(f==null?void 0:f.properties.wind,m.properties.wind,M),type:"forecast",showCircle:p});o.push(N)}D+=t*60,S=h.clone().add(D,"minute").set({minute:0,second:0,millisecond:0})}}return o}static diversionPassageAt(e,t,n,o,i={requestId:""}){var d,a,p,h,D,P,b,S,m;const s=(a=(d=t==null?void 0:t.forecasts)==null?void 0:d[0])==null?void 0:a.hours,r=((p=t==null?void 0:t.history)==null?void 0:p[0])||(s==null?void 0:s[(h=Object.keys(s))==null?void 0:h[0]]);R.info("[%s] the first evasion point: %j",i.requestId,r);const c=(D=Object.keys(s||{}).filter(f=>Number(f)<=24))==null?void 0:D.at(-1),u=s==null?void 0:s[c||-1];if(R.info("[%s] the second evasion point in %s hrs: %j",i.requestId,c,r),r&&u){const f=v.calculateBearing(e,r),M=v.calculateBearing(r,u),x=v.calculateCoordinate(r,f-M+90,n);R.info("[%s] the right tangent point: %j",i.requestId,{from:e,p1:r,p2:u,radius:n,bearing1:f,bearing2:M,right:x});const $=v.calculateBearing(e,x);let N=v.calculateCoordinate(e,$,o*Number(c));N.utc=y((b=(P=t==null?void 0:t.forecasts)==null?void 0:P[0])==null?void 0:b.date).add(Number(c),"hours").utc().format(),R.info("[%s] the temp evasion point: %j",i.requestId,N);const L=v.calculateDistance(N,u);return R.info("[%s] the distance(%d) between tmp & p2: %j - %j",i.requestId,L,N,u),L<n&&(N=v.calculateCoordinate(e,$,o*Number(c)/2),N.utc=y((m=(S=t==null?void 0:t.forecasts)==null?void 0:S[0])==null?void 0:m.date).add(Number(c)/2,"hours").utc().format(),R.warn("[%s] tmp is too close to p2, replace with 1/2 position: %j",i.requestId,N)),N}else{R.info("[%s] no need evasion: %j",i.requestId,{from:e,p1:r,p2:u});return}}static pickIndex(e,t){let n=0;for(const o of e){if(y(o.properties.date).isAfter(t))return n===0?-1:n;n++}return n}static computeNumber(e,t,n){if(e)if(t){if(isNaN(e)&&isNaN(t)&&typeof e!="string"&&typeof t!="string"){const o={};for(const i in e)o[i]=this.computeNumber(e[i],t[i],n);return o}return Math.round((e+(t-e)*n)*100)/100}else return e;else return t}}C.LaneHelper=v,C.LngLatHelper=l,C.TropicalHelper=B,Object.defineProperty(C,Symbol.toStringTag,{value:"Module"})});
1
+ (function(C,R){typeof exports=="object"&&typeof module<"u"?R(exports,require("@turf/turf"),require("moment"),require("moment-timezone"),require("tz-lookup")):typeof define=="function"&&define.amd?define(["exports","@turf/turf","moment","moment-timezone","tz-lookup"],R):(C=typeof globalThis<"u"?globalThis:C||self,R(C["idm-plugin-rabbitmq"]={},C["@turf/turf"],C.moment,C["moment-timezone"],C["tz-lookup"]))})(this,function(C,R,N,B,j){"use strict";function $(P){const e=Object.create(null,{[Symbol.toStringTag]:{value:"Module"}});if(P){for(const t in P)if(t!=="default"){const n=Object.getOwnPropertyDescriptor(P,t);Object.defineProperty(e,t,n.get?n:{enumerable:!0,get:()=>P[t]})}}return e.default=P,Object.freeze(e)}const g=$(R);function k(P){return P&&P.__esModule&&Object.prototype.hasOwnProperty.call(P,"default")?P.default:P}class I{log(){}isLevelEnabled(){return!1}addContext(){}removeContext(){}clearContext(){}}["Trace","Debug","Info","Warn","Error","Fatal","Mark"].forEach(P=>{I.prototype[P.toLowerCase()]=()=>{},I.prototype[`is${P}Enabled`]=()=>!1});const x=(()=>{try{return require("log4js")}catch{return null}})();var L={getLogger:x?x.getLogger:()=>new I};const W=k(L);class l{static guessTimeZoneOffset(e,t){const n=j(t,e),s=N().tz(n).utcOffset();return this.roundPrecision(s/60,1)}static prettyTimeZoneOffset(e){let t=Math.floor(Math.abs(e)),n=Math.round((Math.abs(e)-t)*60);return n=n>9?n:`0${n}`,t=t>9?t:`0${t}`,e>0?`+${t}:${n}`:`-${t}:${n}`}static lng2pretty(e,t=6,n="H°M′"){e=l.convertToStdLng(e,t);let s="E";e<0&&(s="W"),e=Math.abs(e),n=n.toUpperCase();let o=e*3600,i,r,c,a,d,u;i=o%3600%60,n.indexOf("S")!==-1&&(o=o-i,r=l.padNumber(i,2,2)),c=o/60%60,n.indexOf("M")!==-1&&(n.indexOf("S")!==-1?a=l.roundPrecision(c,t).toString().padStart(2,"0"):a=l.padNumber(c,2,2),o=o-c*60),d=o/3600,n.indexOf("M")!==-1?u=l.roundPrecision(d,t).toString().padStart(3,"0"):u=l.padNumber(d,3,2);const p=`${n.replace(/S+/gi,r).replace(/M+/gi,a).replace(/H+/gi,u)}${s}`;return{direction:s,degree:l.roundPrecision(d,t),minute:l.roundPrecision(c,t),second:l.roundPrecision(i,t),pretty:p}}static lat2pretty(e,t=6,n="H°M′"){e=e%180;let s="N";e<0&&(s="S"),e=Math.abs(e),n=n.toUpperCase();let o=e*3600,i,r,c,a,d,u;i=o%3600%60,n.indexOf("S")!==-1&&(o=o-i,r=l.padNumber(i,2,2)),c=o/60%60,n.indexOf("M")!==-1&&(n.indexOf("S")!==-1?a=l.roundPrecision(c,t).toString().padStart(2,"0"):a=l.padNumber(c,2,2),o=o-c*60),d=o/3600,n.indexOf("M")!==-1?u=l.roundPrecision(d,t).toString().padStart(2,"0"):u=l.padNumber(d,2,2);const p=`${n.replace(/S+/gi,r).replace(/M+/gi,a).replace(/H+/gi,u)}${s}`;return{direction:s,degree:l.roundPrecision(d,t),minute:l.roundPrecision(c,t),second:l.roundPrecision(i,t),pretty:p}}static str2Lng(e,t=6){let n;if(isNaN(e)){e=l.strReplace(e,"LNG");const s=e[e.length-1].toUpperCase();e=e.substring(0,e.length-1).trim();const o=e.split(" ").filter(c=>c!=="").map(c=>Number(c));let[i,r]=o;if(r=r>60?r/Math.pow(10,String(r).length-2):r,i>360&&!r){const c=this.roundPrecision(i/100,0);r=i-c*100,i=c}n=i+(r??0)/60,s==="W"&&(n=n*-1)}else n=Number(e);return l.convertToStdLng(n,t)}static str2Lat(e,t=6){let n;if(isNaN(e)){e=l.strReplace(e,"LAT");const s=e[e.length-1].toUpperCase();e=e.substring(0,e.length-1).trim();const o=e.split(" ").filter(c=>c!=="").map(c=>Number(c));let[i,r]=o;if(r=r>60?r/Math.pow(10,String(r).length-2):r,i>90&&!r){const c=this.roundPrecision(i/100,0);r=i-c*100,i=c}n=i+(r??0)/60,s==="S"&&(n=n*-1)}else n=Number(e);return l.roundPrecision(n,t)}static str2LngOrLat(e,t=6,n="LAT"){e=l.strReplace(e,n);const s=e[e.length-1].toUpperCase();return["N","S"].includes(s)?{lat:l.str2Lat(e,t)}:{lng:l.str2Lng(e,t)}}static convertToStdLng(e,t=4){return e>180?(e=e%360,e=e>180?e-360:e):e<-180&&(e=e%360,e=e<-180?e+360:e),l.roundPrecision(e,t)}static roundPrecision(e,t=4){if(typeof e=="number"){const n=Number("1".padEnd(t+1,"0"));return Math.round(e*n)/n}return e}static convertToMonotonicLng2(e){for(let t=1;t<e.length;t++)e[t][0]+=Math.round((e[t-1][0]-e[t][0])/360)*360;return e}static convertToMonotonicLng(e){for(let t=1;t<e.length;t++)e[t].lng+=Math.round((e[t-1].lng-e[t].lng)/360)*360;return e}static strReplace(e,t="LAT"){e=e.replace(/([0-9]+)\.([0-9]+\.[0-9]+)/g,"$1 $2").replace(/-/g," ").replace(/°/," ").replace(/'/g," ").replace(/′/g," ").replace(/"/g," ").replace(/∼/g," ").replace(/°/g," ").replace(/,/g,".").replace(/^ /g,"").replace(/ $/g,"").trim();const n=e[e.length-1].toUpperCase();if(!["N","S","E","W"].includes(n)){const s=e,o=Number(s.split(" ")[0]);if(isNaN(o))throw new Error(`invalid Lat/Lng: ${e}`);o>=90?e=`${s}E`:o<=-90?e=`${s}W`:["LAN","LNG"].includes(t==null?void 0:t.toUpperCase())?e=`${s}${o>0?"E":"W"}`:e=`${s}${o>0?"N":"S"}`}return e}static padNumber(e,t=2,n=2){const s=Math.trunc(e).toString().padStart(t,"0"),o=Math.trunc(l.roundPrecision(e-Math.trunc(e),n)*Math.pow(10,n)).toString().padStart(n,"0");return`${s}.${o}`}}class v{static calculateBearing(e,t,n=!0,s=4){const o=g.points([[e.lng,e.lat],[t.lng,t.lat]]);let i;return n?i=g.rhumbBearing(o.features[0],o.features[1]):i=g.bearing(o.features[0],o.features[1]),i<0&&(i+=360),l.roundPrecision(i,s)}static calculateDistance(e,t,n=!0,s=4,o="nauticalmiles"){e={...e},t={...t},e.lng=l.convertToStdLng(e.lng,s),t.lng=l.convertToStdLng(t.lng,s);const i=g.points([[e.lng,e.lat],[t.lng,t.lat]]);let r;return n?r=g.rhumbDistance(i.features[0],i.features[1],{units:o}):r=g.distance(i.features[0],i.features[1],{units:o}),l.roundPrecision(r,s)}static calculateRouteDistance(e,t=4,n="nauticalmiles"){let s=0,o;for(const i of e)for(let r=0;r<i.length-1;r++){const c={lng:i[r][0],lat:i[r][1]};r===0&&o&&(s+=this.calculateDistance(o,c,!0,t,n));const a={lng:i[r+1][0],lat:i[r+1][1]};s+=this.calculateDistance(c,a,!0,t,n),o=a}return l.roundPrecision(s,t)}static calculateCoordinate(e,t,n,s="nauticalmiles",o=!0){const i=g.point([e.lng,e.lat]);let r;o?r=g.rhumbDestination(i,n,t,{units:s}):r=g.destination(i,n,t,{units:s});const c=r.geometry.coordinates;return{lng:l.convertToStdLng(c[0],8),lat:l.roundPrecision(c[1],8)}}static interpolateCoordinates(e,t,n,s=!0,o=!0,i="nauticalmiles"){const r=[],c=this.calculateBearing(e,t,!1),a=this.calculateDistance(e,t,!1,8,i);s&&r.push({lng:e.lng,lat:e.lat});let d=0;for(;d<a;)d+=n,d<a&&r.push(this.calculateCoordinate(e,c,d,i,!1));return o&&r.push({lng:t.lng,lat:t.lat}),r}static divideAccordingToLng(e,t=!1){if((e==null?void 0:e.length)<2)return[];e=this.deduplicateCoordinates(e);let n=[];const s=[];let o,i;for(let r=0;r<e.length-1;r++){o=l.convertToStdLng(e[r].lng,8),i=l.convertToStdLng(e[r+1].lng,8),n.push([o,e[r].lat]);const c=o-i;if(Math.abs(c)>180){const a=l.convertToMonotonicLng2([[o,e[r].lat],[i,e[r+1].lat]]);let d,u;t?(d=g.lineString(a),u=g.lineString([[c>0?180:-180,89],[c>0?180:-180,-89]])):(d=g.greatCircle(a[0],a[1]),u=g.greatCircle([c>0?180:-180,89],[c>0?180:-180,-89]));const p=g.lineIntersect(d,u);let f;if(p.features.length){const M=g.getCoord(p.features[0]);f=l.roundPrecision(M[1],8)}else f=e[r].lat;c>0?(n.push([180-1e-6,f]),s.push([...n]),n=[],n.push([-(180-1e-6),f])):(n.push([-(180-1e-6),f]),s.push([...n]),n=[],n.push([180-1e-6,f]))}r===e.length-2&&n.push([i,e[r+1].lat])}return s.push(n),s}static deduplicateRoute(e){const t=[];for(const n of e){const s=n.reduce((o,i)=>(o.findIndex(r=>r[0]===i[0]&&r[1]===i[1])===-1&&o.push(i),o),[]);t.push(s)}return t}static deduplicateCoordinates(e){return e.reduce((t,n)=>(t.findIndex(s=>s.lat===n.lat&&s.lng===n.lng)===-1&&t.push(n),t),[])}static removeCoordinateFromRoute(e,t){e.lng=l.convertToStdLng(e.lng,8);for(const n of t)for(let s=n.length-1;s>=0;s--)l.roundPrecision(n[s][0],8)===e.lng&&l.roundPrecision(n[s][1],8)===l.roundPrecision(e.lat,8)&&n.splice(s,1);return t}static removeCoordinateFromWaypoints(e,t){e.lng=l.convertToStdLng(e.lng,8);for(let n=t.length-1;n>=0;n--)l.roundPrecision(t[n].lng,8)===e.lng&&l.roundPrecision(t[n].lat,8)===l.roundPrecision(e.lat,8)&&t.splice(n,1);return t}static mergeCoordinateToRoute(e,t){e.lng=l.convertToStdLng(e.lng,8);let n=Number.MAX_VALUE,s=0,o=0,i,r;return t.forEach((c,a)=>{for(let d=0;d<c.length-1;d++){const u={lng:c[d][0],lat:c[d][1]},p={lng:c[d+1][0],lat:c[d+1][1]},f=this.calculatePointToLineDistance(e,u,p);n>f&&(n=f,o=d,s=a,i=this.calculateDistance(u,e),r=this.calculateDistance(p,e))}}),i!==0&&r!==0?t[s].splice(o+1,0,[e.lng,e.lat]):i===0?t[s].splice(o,1,[e.lng,e.lat]):r===0&&t[s].splice(o+1,1,[e.lng,e.lat]),t}static appendCoordinateToRoute(e,t){e.lng=l.convertToStdLng(e.lng,8);const n=v.convertRouteToCoordinates(t);return n.push(e),v.divideAccordingToLng(n)}static unshiftCoordinateToRoute(e,t){const n=v.convertRouteToCoordinates(t);return n.unshift(e),v.divideAccordingToLng(n)}static mergeWaypointsToRoute(e,t){for(const n of e)t=this.mergeCoordinateToRoute(n,t);return t}static calculateRangeRoute(e,t,n){n=this.mergeWaypointsToRoute([e,t],n);const s=[];let o=0;return n.forEach(i=>{if(o===2)return;const r=[];for(const c of i){if(l.roundPrecision(t.lng,8)===l.roundPrecision(c[0],8)&&l.roundPrecision(t.lat,8)===l.roundPrecision(c[1],8)){r.push(c),o===0&&r.push([e.lng,e.lat]),o=2;break}o===1?r.push(c):l.roundPrecision(e.lng,8)===l.roundPrecision(c[0],8)&&l.roundPrecision(e.lat,8)===l.roundPrecision(c[1],8)&&(o=1,r.push(c))}r.length&&s.push(r)}),s}static calculateRangeWaypoints(e,t,n,s=[]){const o=this.convertRouteToCoordinates(n,0),i=this.mergeCoordinatesToWaypoints([e,t,...s],o),r=i.findIndex(d=>l.roundPrecision(e.lng,8)===l.roundPrecision(d.lng,8)&&l.roundPrecision(e.lat,8)===l.roundPrecision(d.lat,8)),c=i.findIndex(d=>l.roundPrecision(t.lng,8)===l.roundPrecision(d.lng,8)&&l.roundPrecision(t.lat,8)===l.roundPrecision(d.lat,8));return i.filter((d,u)=>u>=r&&u<=c)}static calculateMinDistanceToRoute(e,t){let n=Number.MAX_VALUE,s=0,o=0;return t.forEach((i,r)=>{for(let c=0;c<i.length-1;c++){const a={lng:i[c][0],lat:i[c][1]},d={lng:i[c+1][0],lat:i[c+1][1]},u=this.calculatePointToLineDistance(e,a,d);n>u&&(n=u,s=c,o=r)}}),{minDist:n,segIndex:o,minIndex:s}}static calculateSubRoute(e,t){const{segIndex:n,minIndex:s}=this.calculateMinDistanceToRoute({...e},t);e.lng=l.convertToStdLng(e.lng);const o=[];let i=!0;for(let r=n;r<t.length;r++)if(i){const c=[];c.push([e.lng,e.lat]);for(let a=s+1;a<t[r].length;a++)e.lng===t[r][a][0]&&e.lat===t[r][a][1]||c.push(t[r][a]);o.push(c),i=!1}else o.push([...t[r]]);return o}static calculateSubWaypoints(e,t){let n=Number.MAX_VALUE,s=0;for(let i=0;i<t.length-1;i++){const r=t[i],c=t[i+1];if(this.calculateDistance(e,r)===0)return t;if(this.calculateDistance(e,c)===0)return t.filter((d,u)=>u>0);const a=this.calculatePointToLineDistance(e,r,c);n>a&&(n=a,s=i)}e.lng=l.convertToStdLng(e.lng);const o=[e];for(let i=s+1;i<t.length;i++)o.push(t[i]);return o}static calculatePointToLineDistance(e,t,n,s={units:"nauticalmiles",method:"geodesic"}){e.lng=l.convertToStdLng(e.lng),t={...t},n={...n},t.lng=l.convertToStdLng(t.lng,8),n.lng=l.convertToStdLng(n.lng,8);const o=l.convertToMonotonicLng([t,n]);t=o[0],n=o[1];const i=g.lineString([[t.lng,t.lat],[n.lng,n.lat]]),r=g.pointToLineDistance(g.point([e.lng,e.lat]),i,s),c=g.pointToLineDistance(g.point([e.lng>0?e.lng-360:e.lng+360,e.lat]),i,s);return l.roundPrecision(Math.min(r,c),6)}static calculateWaypointsPropInRoute(e,t){t=this.mergeWaypointsToRoute(e,t);for(let n=0;n<e.length-1;n++){const s=e[n],o=e[n+1],i=this.calculateRangeRoute(s,o,t);n===0&&(s.distanceFromPrevious=0,s.distanceFromStart=0),o.distanceFromPrevious=this.calculateRouteDistance(i),o.distanceFromStart=l.roundPrecision((s.distanceFromStart||0)+o.distanceFromPrevious)}return e}static mergeCoordinatesToWaypoints(e,t,n=!0){for(const s of e)this.mergeCoordinateToWaypoints(s,t,n);return t}static mergeCoordinateToWaypoints(e,t,n=!0){e.lng=l.convertToStdLng(e.lng,8);let s=Number.MAX_VALUE,o=0,i=0,r=0;for(let c=0;c<t.length-1;c++){const a={lng:t[c].lng,lat:t[c].lat},d={lng:t[c+1].lng,lat:t[c+1].lat},u=this.calculatePointToLineDistance(e,a,d);s>=u&&(s=u,o=c,i=this.calculateDistance(a,e,!1,6),r=this.calculateDistance(d,e,!1,6))}return i!==0&&r!==0?i<s||i===s&&o===0?t.unshift(e):r<s||r===s&&o===t.length-2?t.push(e):t.splice(o+1,0,e):i===0?n&&t.splice(o,1,e):r===0&&n&&t.splice(o+1,1,e),t.map(c=>(c.lng=l.convertToStdLng(c.lng),c))}static generateRouteAccordingToWaypoints(e){const t=[];for(let n=1;n<e.length;n++){const s=e[n-1],o=e[n];if(n===1&&t.push(s),o.gcToPrevious){const i=this.interpolateCoordinates(s,o,200,!1,!0,"nauticalmiles");t.push(...i)}else t.push(o)}return this.divideAccordingToLng(t,!0)}static nearestCoordinateInRoute(e,t){const n=g.point([e.lng,e.lat]),s=[];for(const c of t){const a=c.map(d=>g.point(d));s.push(...a)}const o=g.featureCollection(s),i=g.nearestPoint(n,o),r=g.getCoord(i);return{lng:r[0],lat:r[1]}}static calculatePrevWaypoint(e,t){let n=0;this.mergeCoordinateToWaypoints(e,t);for(let s=0;s<t.length-1;s++){const o=t[s],i=t[s+1];if(this.calculateDistance(e,o)===0){n=s;break}if(this.calculateDistance(e,i)===0){n=s+1;break}}return t[n===0?0:n-1]}static calculateNextCoordinateAlongRoute(e,t,n,s="nauticalmiles"){var p;const o=e.speed||12,i=[];let r=[],c=!1,a=0,d=0,u;if(t&&n.length?(i.push(e),n.forEach((f,M)=>{if(c)r.push(f);else{const S=[];let b;for(let T=0;T<f.length;T++)if(u)S.push(f[T]);else{b={lng:f[T][0],lat:f[T][1]};const m=this.calculateDistance(e,b,!0,8,s);if(a+=m,a<t)d+=m,i.push(b),e=b;else{if(d=t,a===t)u=b,S.push([u.lng,u.lat]);else{const h=a-t,D=this.calculateBearing(b,e);u=this.calculateCoordinate(b,D,h,s),S.push([u.lng,u.lat]),S.push([b.lng,b.lat])}c=!0}}S.length&&r.push(S),M===n.length-1&&!u&&(u=b)}})):(r=n,u={...e}),u)if(i.push(u),u.distanceFromPrevious=d,u.hourFromPrevious=Math.round(d/o*1e4)/1e4,((p=r[0])==null?void 0:p.length)>1){const f={lng:r[0][1][0],lat:r[0][1][1]};u.bearing=this.calculateBearing(u,f)}else u.bearing=0;return{coordinate:u,nextRoute:r,prevRoute:i}}static nearestCoordinateInLine(e,t,n){const s=l.convertToStdLng(e.lng,6),o=g.point([s,e.lat]),i=l.convertToStdLng(t.lng,6),r=l.convertToStdLng(n.lng,6),c=g.lineString([[i,t.lat],[r,n.lat]]),a=g.nearestPointOnLine(c,o),d=g.getCoord(a),u=l.roundPrecision(d[0],6),p=l.roundPrecision(d[1],6);return{lng:u,lat:p,inline:!(u===i&&p===t.lat)&&!(u===r&&p===n.lat)}}static convertRouteToCoordinates(e,t=0){const n=[];let s,o;return e.forEach(i=>{i.forEach(r=>{const c={lng:r[0],lat:r[1]};if(!o)n.push(c),o=c;else if(o.bearing===void 0)o.bearing=this.calculateBearing(o,c,!0);else{const a=this.calculateDistance(s,c,!0);a&&a>=t&&(s.bearing=this.calculateBearing(s,c,!0),n.push(s),o=s)}s=c})}),s&&n.push(s),n}static simplifyRouteToCoordinates(e,t,n=1){let s=this.convertRouteToCoordinates(e,n);return s=this.simplifyGCCoordinates(s,t),s}static simplifyGCCoordinates(e,t){t.forEach(s=>{this.mergeCoordinateToWaypoints(s,e)});for(let s=1;s<t.length;s++){const o=t[s-1],i=t[s];if(i.gcToPrevious){const r=e.findIndex(a=>a.lng===o.lng&&a.lat===o.lat),c=e.findIndex(a=>a.lng===i.lng&&a.lat===i.lat);for(let a=c-1;a>r;a--)e.splice(a,1)}}let n=0;for(let s=1;s<e.length;s++){const o=e[s-1],i=e[s];i.gcToPrevious?(o.bearing=this.calculateBearing(o,i,!1),i.distanceFromPrevious=this.calculateDistance(o,i,!1)):(o.bearing=this.calculateBearing(o,i,!0),i.distanceFromPrevious=this.calculateDistance(o,i,!0)),n=l.roundPrecision(n+i.distanceFromPrevious),i.distanceFromStart=n}return e.map(s=>(s.lng=l.convertToStdLng(s.lng),s))}static calculateCenter(e){const t=[];for(const r of e)for(const c of r)t.push(c);const n=g.featureCollection([]),s=l.convertToMonotonicLng2(t);for(const r of s)n.features.push(g.point(r));const i=g.center(n).geometry.coordinates;return{lng:l.convertToStdLng(i[0],8),lat:l.roundPrecision(i[1],8)}}static calculateCenter2(e){const t=this.generateRouteAccordingToWaypoints(e);return this.calculateCenter(t)}static calculateBBox(e){const t=[];for(const o of e)for(const i of o)t.push(i);const n=l.convertToMonotonicLng2(t),s=g.lineString(n);return g.bbox(s)}static calculateBBox2(e){const t=this.generateRouteAccordingToWaypoints(e);return this.calculateBBox(t)}}const y=W.getLogger("vessel");class O{static convert2Geojson(e){var n;const t=g.featureCollection([]);for(const s of e){if(s.forecasts){const o=(n=s.history)==null?void 0:n[0];for(const i of s.forecasts){const r=[],c=N(i.date).utc(),a=`${s.name}-${i.model}`;if(o){const d=N(o.updated).utc(),u=g.point([o.lng,o.lat],{model:i.model,name:s.name,date:d.format(),hour:0,format:d.format("MMM-DD/HHmm[Z]"),pressure:o.pressure>1e4?l.roundPrecision(o.pressure/100,0):l.roundPrecision(o.pressure,0),wind:{kts:o.kts,spd:o.speed||o.spd},category:a,type:"forecast"});t.features.push(u),r.push(u.geometry.coordinates)}for(const d in i==null?void 0:i.hours){const u=i.hours[d];u.wind.spd=u.wind.spd||u.wind.speed;const p=c.clone().add(Number(d),"hour"),f=g.point([u.lng,u.lat],{model:i.model,name:s.name,date:p.format(),hour:Number(d),format:p.format("MMM-DD/HHmm[Z]"),pressure:u.pressure>1e4?l.roundPrecision(u.pressure/100,0):l.roundPrecision(u.pressure,0),gusts:u.gusts,wind:u.wind||{},movement:u.movement,category:a,type:"forecast"});t.features.push(f),r.push(f.geometry.coordinates)}if((r==null?void 0:r.length)>1){const d=g.lineString(l.convertToMonotonicLng2(r),{date:i.date,id:s.id||s.name,model:i.model,name:s.name,category:a,type:"forecast"});t.features.push(d)}}}if(s.history){const o=[];for(const r of s.history){const c=N(r.updated).utc(),a=g.point([r.lng,r.lat],{name:s.name,date:c.format(),format:c.format("MMM-DD/HHmm[Z]"),pressure:r.pressure>1e4?l.roundPrecision(r.pressure/100,0):l.roundPrecision(r.pressure,0),spd:r.speed||r.spd,kts:r.kts,source:r.source,level:r.type,type:"history",category:`${s.name}-history`});t.features.push(a),o.push(a.geometry.coordinates)}const i=s.history[0];if(o.length===1&&o.push(o[0]),o.length>1){const r=g.lineString(l.convertToMonotonicLng2(o),{name:s.name,type:"history",updated:i==null?void 0:i.updated,pressure:(i==null?void 0:i.pressure)>1e4?l.roundPrecision((i==null?void 0:i.pressure)/100,0):l.roundPrecision(i==null?void 0:i.pressure,0),spd:(i==null?void 0:i.speed)||(i==null?void 0:i.spd),kts:i==null?void 0:i.kts,source:i==null?void 0:i.source,level:i==null?void 0:i.type});t.features.push(r)}}}return t}static interpolate(e,t=3){var o,i,r,c;const n=(o=e==null?void 0:e.data)==null?void 0:o.features.filter(a=>a.geometry.type==="LineString"&&a.properties.type==="forecast"),s=[];for(const a of n){const d=a.properties.name,u=a.properties.model,p=a.properties.showCircle,f=N(a.properties.date).utc();let M=t*60-(f.get("hour")*60+f.get("minute"))%(t*60);const S=(i=e==null?void 0:e.data)==null?void 0:i.features.filter(m=>m.geometry.type==="Point"&&m.properties.type==="forecast"&&m.properties.category===`${d}-${u}`);let b,T=f.clone().add(M,"minute").set({minute:0,second:0,millisecond:0});for(;b=this.pickIndex(S,T),b<=S.length-1;){if(b>0){const m=S[b],h=b===0?void 0:S[b-1],D=(M/60-((r=h==null?void 0:h.properties)==null?void 0:r.hour))/(m.properties.hour-((c=h==null?void 0:h.properties)==null?void 0:c.hour)),E=this.computeNumber(h==null?void 0:h.geometry.coordinates[0],m.geometry.coordinates[0],D),F=this.computeNumber(h==null?void 0:h.geometry.coordinates[1],m.geometry.coordinates[1],D),A=g.point([E,F],{name:d,model:u,category:m==null?void 0:m.properties.category,date:T.format(),format:T.format("MMM-DD/HHmm[Z]"),gusts:this.computeNumber(h==null?void 0:h.properties.gusts,m.properties.gusts,D),hour:this.computeNumber(h==null?void 0:h.properties.hour,m.properties.hour,D),movement:this.computeNumber(h==null?void 0:h.properties.movement,m.properties.movement,D),pressure:this.computeNumber(h==null?void 0:h.properties.pressure,m.properties.pressure,D),wind:this.computeNumber(h==null?void 0:h.properties.wind,m.properties.wind,D),type:"forecast",showCircle:p});s.push(A)}M+=t*60,T=f.clone().add(M,"minute").set({minute:0,second:0,millisecond:0})}}return s}static diversionPassageAt(e,t,n,s,o={requestId:""}){var a,d,u,p;const{t1:i,t2:r,hr:c}=this.tropicalCenterTwin(t,24,o);if(i&&r){const f=v.calculateBearing(e,i),M=v.calculateBearing(i,r),S=v.calculateCoordinate(i,f-M+90,n);y.info("[%s] the right tangent position: %j",o.requestId,{from:e,t1:i,t2:r,radius:n,bearing1:f,bearing2:M,right:S});const b=v.calculateBearing(e,S);let T=v.calculateCoordinate(e,b,s*Number(c));T.utc=N((d=(a=t==null?void 0:t.forecasts)==null?void 0:a[0])==null?void 0:d.date).add(Number(c),"hours").utc().format(),y.info("[%s] the diversion position: %j",o.requestId,T);const m=v.calculateDistance(T,r);return y.info("[%s] the distance(%d) between the diversion position & t2: %j - %j",o.requestId,m,T,r),m<n&&(T=v.calculateCoordinate(e,b,s*Number(c)/2),T.utc=N((p=(u=t==null?void 0:t.forecasts)==null?void 0:u[0])==null?void 0:p.date).add(Number(c)/2,"hours").utc().format(),y.warn("[%s] the prev diversion position is too close to t2, replace with 1/2 position: %j",o.requestId,T)),T}else{y.info("[%s] no need to diversion: %j",o.requestId,{from:e,t1:i,t2:r});return}}static driftPassageAt(e,t,n,s={requestId:""}){const{t1:o,t2:i,hr:r}=this.tropicalCenterTwin(t,24,s);if(o&&i){const c=v.calculateBearing(e,o),a=v.calculateBearing(o,i);return v.calculateCoordinate(o,c-a+180,n)}else{y.info("[%s] no need drift: %j",s.requestId,{from:e,t1:o,t2:i});return}}static tropicalCenterTwin(e,t=24,n={requestId:""}){var c,a,d,u,p;let s={};(c=e.forecasts)==null||c.forEach(f=>{s={...f.hours,...s}});const o=((a=e==null?void 0:e.history)==null?void 0:a[0])||(s==null?void 0:s[(d=Object.keys(s))==null?void 0:d[0]]);y.info("[%s] the first tropical center: %j",n.requestId,o);let i=(u=Object.keys(s||{}).filter(f=>Number(f)<=(t<0?24:t)))==null?void 0:u.at(-1);i||(i=(p=Object.keys(s||{}).filter(f=>Number(f)<=(t<0?24:2*t)))==null?void 0:p.at(-1));const r=s==null?void 0:s[i||-1];return y.info("[%s] the second tropical center: %j in %d hrs",n.requestId,r,i),{t1:o,t2:r,hr:i}}static pickIndex(e,t){let n=0;for(const s of e){if(N(s.properties.date).isAfter(t))return n===0?-1:n;n++}return n}static computeNumber(e,t,n){if(e)if(t){if(isNaN(e)&&isNaN(t)&&typeof e!="string"&&typeof t!="string"){const s={};for(const o in e)s[o]=this.computeNumber(e[o],t[o],n);return s}return Math.round((e+(t-e)*n)*100)/100}else return e;else return t}}C.LaneHelper=v,C.LngLatHelper=l,C.TropicalHelper=O,Object.defineProperty(C,Symbol.toStringTag,{value:"Module"})});
@@ -30,11 +30,14 @@ export declare class TropicalHelper {
30
30
  showCircle: any;
31
31
  }>[];
32
32
  /**
33
- * 航行方向与台风
34
- * @param from
35
- * @param tropical
36
- * @param radius
37
- * @param speed
33
+ * 计算最佳绕航点
34
+ * 1) 确定当前船位与当前台风中心点之间的方位角,
35
+ * 2) 当前台风中心点与近24小时台风中心点的方位角,
36
+ * 3) 根据1)、2)及可航半圆计算绕航点
37
+ * @param from 船舶位置
38
+ * @param tropical 台风数据 { history: any[], forecasts: any[] }
39
+ * @param radius 与台风中心的距离
40
+ * @param speed 前进速度
38
41
  * @param options
39
42
  */
40
43
  static diversionPassageAt(from: {
@@ -43,6 +46,33 @@ export declare class TropicalHelper {
43
46
  }, tropical: any, radius: number, speed: number, options?: {
44
47
  requestId: string;
45
48
  }): any;
49
+ /**
50
+ * 计算最佳漂航点
51
+ * 1) 确定当前船位与当前台风中心点之间的方位角,
52
+ * 2) 当前台风中心点与近24小时台风中心点的方位角,
53
+ * 3) 根据2)、3)反向计算航漂航点
54
+ * @param from 船舶位置
55
+ * @param tropical 台风数据 { history: any[], forecasts: any[] }
56
+ * @param radius 与台风中心的距离
57
+ * @param options
58
+ */
59
+ static driftPassageAt(from: {
60
+ lat: number;
61
+ lng: number;
62
+ }, tropical: any, radius: number, options?: {
63
+ requestId: string;
64
+ }): {
65
+ lng: number;
66
+ lat: number;
67
+ } | undefined;
68
+ /**
69
+ * 获取台风中心点对
70
+ * @param tropical { history: any[], forecasts: any[] }
71
+ * @param hrs 未来小时数,默认24
72
+ * @param options
73
+ * @private
74
+ */
75
+ private static tropicalCenterTwin;
46
76
  private static pickIndex;
47
77
  private static computeNumber;
48
78
  }
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@idm-plugin/geo",
3
3
  "private": false,
4
- "version": "1.2.2",
4
+ "version": "1.2.4",
5
5
  "description": "idm plugin for geo",
6
6
  "type": "module",
7
7
  "keywords": [