2023-03-14 22:08:30 +00:00
|
|
|
/* Thanks to pinsafe from BangleApps repository */
|
|
|
|
|
|
|
|
var Layout = require("Layout");
|
2023-11-17 03:09:45 +00:00
|
|
|
const BANGLEJS2 = process.env.HWVERSION == 2; // check for bangle 2
|
2023-03-14 22:08:30 +00:00
|
|
|
|
|
|
|
const W = g.getWidth();
|
|
|
|
const H = g.getHeight();
|
|
|
|
|
|
|
|
var wp = require('Storage').readJSON("waypoints.json", true) || [];
|
|
|
|
// Use this with corrupted waypoints
|
|
|
|
//var wp = [];
|
|
|
|
/* 0 .. DD.ddddd
|
|
|
|
1 .. DD MM.mmm'
|
|
|
|
2 .. DD MM'ss"
|
|
|
|
*/
|
|
|
|
var mode = 1;
|
2023-10-12 12:30:30 +00:00
|
|
|
var key; /* Shared between functions, typically wp name */
|
|
|
|
var fix; /* GPS fix */
|
|
|
|
var cancel_gps;
|
|
|
|
var gps_start;
|
2023-03-14 22:08:30 +00:00
|
|
|
|
|
|
|
function writeWP() {
|
|
|
|
require('Storage').writeJSON("waypoints.json", wp);
|
|
|
|
}
|
|
|
|
|
|
|
|
function mainMenu() {
|
2023-11-16 20:39:37 +00:00
|
|
|
let textInputInstalled = true;
|
|
|
|
try {
|
|
|
|
require("textinput")
|
|
|
|
} catch(err) {
|
|
|
|
textInputInstalled = false;
|
|
|
|
}
|
2023-03-14 22:08:30 +00:00
|
|
|
var menu = {
|
2023-11-16 20:39:37 +00:00
|
|
|
"< Back" : () => load()
|
2023-03-14 22:08:30 +00:00
|
|
|
};
|
2023-11-16 20:39:37 +00:00
|
|
|
for (let id in wp) {
|
2023-03-14 22:08:30 +00:00
|
|
|
let i = id;
|
2023-11-03 20:56:10 +00:00
|
|
|
menu[wp[id]["name"]]=()=>{ show(i); };
|
2023-03-14 22:08:30 +00:00
|
|
|
}
|
2023-11-17 03:09:45 +00:00
|
|
|
if (textInputInstalled && BANGLEJS2) {
|
2023-11-16 20:39:37 +00:00
|
|
|
menu["Add"]=addCard;
|
|
|
|
}
|
2023-03-14 22:08:30 +00:00
|
|
|
menu["Remove"]=removeCard;
|
|
|
|
menu["Format"]=setFormat;
|
2023-11-16 20:39:37 +00:00
|
|
|
if (textInputInstalled) {
|
|
|
|
menu["Mark GPS"]=markGps;
|
|
|
|
}
|
2023-03-14 22:08:30 +00:00
|
|
|
g.clear();
|
|
|
|
E.showMenu(menu);
|
|
|
|
}
|
|
|
|
|
2023-10-12 12:30:30 +00:00
|
|
|
function updateGps() {
|
|
|
|
let have = false, lat = "lat", lon = "lon", alt = "alt", speed = "speed";
|
|
|
|
|
|
|
|
if (cancel_gps)
|
|
|
|
return;
|
|
|
|
fix = Bangle.getGPSFix();
|
|
|
|
|
2023-11-03 21:03:44 +00:00
|
|
|
speed = "no fix for " + (getTime() - gps_start).toFixed(0) + "s";
|
2023-10-12 12:30:30 +00:00
|
|
|
|
|
|
|
if (fix && fix.fix && fix.lat) {
|
2023-11-03 21:03:44 +00:00
|
|
|
lat = "" + lat(fix.lat);
|
|
|
|
lon = "" + lon(fix.lon);
|
|
|
|
alt = "alt " + fix.alt.toFixed(0) + "m";
|
|
|
|
speed = "speed " + fix.speed.toFixed(1) + "kt";
|
2023-10-12 12:30:30 +00:00
|
|
|
have = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
g.reset().setFont("Vector", 20)
|
|
|
|
.setColor(1,1,1)
|
|
|
|
.fillRect(0, 0, 176, 120)
|
|
|
|
.setColor(0,0,0)
|
|
|
|
.drawString(key, 0, 0)
|
|
|
|
.drawString(lat, 0, 20)
|
|
|
|
.drawString(lon, 0, 40)
|
|
|
|
.drawString(alt, 0, 60)
|
|
|
|
.drawString(speed, 0, 80);
|
|
|
|
|
|
|
|
setTimeout(updateGps, 100);
|
|
|
|
}
|
|
|
|
|
|
|
|
function stopGps() {
|
|
|
|
cancel_gps=true;
|
|
|
|
Bangle.setGPSPower(0, "waypoint_editor");
|
|
|
|
}
|
|
|
|
|
2023-11-01 10:36:59 +00:00
|
|
|
function confirmGps(s) {
|
|
|
|
key = s;
|
2023-11-03 20:56:10 +00:00
|
|
|
var la = new Layout (
|
|
|
|
{type:"v", c: [
|
2023-11-03 21:03:44 +00:00
|
|
|
{type:"txt", font:"15%", pad:1, fillx:1, filly:1, label:""},
|
2023-11-03 20:56:10 +00:00
|
|
|
{type:"txt", font:"15%", pad:1, fillx:1, filly:1, label:""},
|
|
|
|
{type:"txt", font:"15%", pad:1, fillx:1, filly:1, label:""},
|
|
|
|
{type:"h", c: [
|
2023-11-03 21:19:08 +00:00
|
|
|
{type:"btn", font:"15%", pad:1, fillx:1, filly:1, label: "YES", cb:l=>{
|
|
|
|
print("should mark", key, fix); createWP(fix.lat, fix.lon, key); cancel_gps=true; mainMenu();
|
|
|
|
}},
|
|
|
|
{type:"btn", font:"15%", pad:1, fillx:1, filly:1, label: " NO", cb:l=>{ cancel_gps=true; mainMenu(); }}
|
2023-11-03 20:56:10 +00:00
|
|
|
]}
|
|
|
|
], lazy:true});
|
2023-10-12 12:30:30 +00:00
|
|
|
g.clear();
|
|
|
|
la.render();
|
|
|
|
updateGps();
|
|
|
|
}
|
|
|
|
|
|
|
|
function markGps() {
|
|
|
|
cancel_gps = false;
|
|
|
|
Bangle.setGPSPower(1, "waypoint_editor");
|
|
|
|
gps_start = getTime();
|
2023-11-01 10:36:59 +00:00
|
|
|
require("textinput").input({text:"wp"}).then(key => {
|
|
|
|
confirmGps(key);
|
|
|
|
});
|
2023-10-12 12:30:30 +00:00
|
|
|
}
|
|
|
|
|
2023-03-14 22:08:30 +00:00
|
|
|
function setFormat() {
|
2023-10-12 12:30:30 +00:00
|
|
|
var la = new Layout (
|
2023-11-03 20:56:10 +00:00
|
|
|
{type:"v", c: [
|
|
|
|
{type:"txt", font:"15%", pad:1, fillx:1, filly:1, label:"Format"},
|
|
|
|
{type:"btn", font:"15%", pad:1, fillx:1, filly:1, label: "DD.dddd", cb:l=>{ mode = 0; mainMenu(); }},
|
|
|
|
{type:"btn", font:"15%", pad:1, fillx:1, filly:1, label: "DD MM.mmm'", cb:l=>{ mode = 1; mainMenu(); }},
|
|
|
|
{type:"btn", font:"15%", pad:1, fillx:1, filly:1, label: "DD MM'ss"+'"', cb:l=>{ mode = 2; mainMenu(); }},
|
|
|
|
], lazy:true});
|
2023-03-14 22:08:30 +00:00
|
|
|
g.clear();
|
2023-10-12 12:30:30 +00:00
|
|
|
la.render();
|
2023-03-14 22:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function format(x) {
|
|
|
|
switch (mode) {
|
|
|
|
case 0:
|
|
|
|
return "" + x;
|
|
|
|
case 1:
|
|
|
|
d = Math.floor(x);
|
|
|
|
m = x - d;
|
|
|
|
m = m*60;
|
|
|
|
return "" + d + " " + m + "'";
|
|
|
|
case 2:
|
|
|
|
d = Math.floor(x);
|
|
|
|
m = x - d;
|
|
|
|
m = m*60;
|
|
|
|
mf = Math.floor(m);
|
|
|
|
s = m - mf;
|
|
|
|
s = s*60;
|
|
|
|
return "" + d + " " + mf + "'" + s + '"';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
function lat(x) {
|
|
|
|
c = "N";
|
|
|
|
if (x<0) {
|
|
|
|
c = "S";
|
|
|
|
x = -x;
|
|
|
|
}
|
|
|
|
return c+format(x);
|
|
|
|
}
|
|
|
|
function lon(x) {
|
|
|
|
c = "E";
|
|
|
|
if (x<0) {
|
|
|
|
c = "W";
|
|
|
|
x = -x;
|
|
|
|
}
|
|
|
|
return c+format(x);
|
|
|
|
}
|
|
|
|
|
2023-11-03 20:56:10 +00:00
|
|
|
function show(pin) {
|
|
|
|
var i = wp[pin];
|
2023-11-16 20:39:37 +00:00
|
|
|
var l = lat(i["lat"]) + "\n" + lon(i["lon"]);
|
|
|
|
E.showPrompt(l,{
|
|
|
|
title:i["name"],
|
|
|
|
buttons : {"Ok":true}
|
|
|
|
}).then(function(v) {
|
|
|
|
mainMenu();
|
|
|
|
});
|
2023-03-14 22:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function showNumpad(text, key_, callback) {
|
|
|
|
key = key_;
|
|
|
|
E.showMenu();
|
|
|
|
function addDigit(digit) {
|
|
|
|
key+=digit;
|
|
|
|
if (1) {
|
|
|
|
l = text[key.length];
|
|
|
|
switch (l) {
|
|
|
|
case '.': case ' ': case "'":
|
|
|
|
key+=l;
|
|
|
|
break;
|
|
|
|
case 'd': case 'D': default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Bangle.buzz(20);
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
function update() {
|
|
|
|
g.reset();
|
|
|
|
g.clearRect(0,0,g.getWidth(),23);
|
|
|
|
s = key + text.substr(key.length, 999);
|
|
|
|
g.setFont("Vector:24").setFontAlign(1,0).drawString(s,g.getWidth(),12);
|
|
|
|
}
|
|
|
|
ds="12%";
|
|
|
|
var numPad = new Layout ({
|
|
|
|
type:"v", c: [{
|
|
|
|
type:"v", c: [
|
|
|
|
{type:"", height:24},
|
|
|
|
{type:"h",filly:1, c: [
|
|
|
|
{type:"btn", font:ds, width:58, label:"7", cb:l=>{addDigit("7");}},
|
|
|
|
{type:"btn", font:ds, width:58, label:"8", cb:l=>{addDigit("8");}},
|
|
|
|
{type:"btn", font:ds, width:58, label:"9", cb:l=>{addDigit("9");}}
|
|
|
|
]},
|
|
|
|
{type:"h",filly:1, c: [
|
|
|
|
{type:"btn", font:ds, width:58, label:"4", cb:l=>{addDigit("4");}},
|
|
|
|
{type:"btn", font:ds, width:58, label:"5", cb:l=>{addDigit("5");}},
|
|
|
|
{type:"btn", font:ds, width:58, label:"6", cb:l=>{addDigit("6");}}
|
|
|
|
]},
|
|
|
|
{type:"h",filly:1, c: [
|
|
|
|
{type:"btn", font:ds, width:58, label:"1", cb:l=>{addDigit("1");}},
|
|
|
|
{type:"btn", font:ds, width:58, label:"2", cb:l=>{addDigit("2");}},
|
|
|
|
{type:"btn", font:ds, width:58, label:"3", cb:l=>{addDigit("3");}}
|
|
|
|
]},
|
|
|
|
{type:"h",filly:1, c: [
|
|
|
|
{type:"btn", font:ds, width:58, label:"0", cb:l=>{addDigit("0");}},
|
|
|
|
{type:"btn", font:ds, width:58, label:"C", cb:l=>{key=key.slice(0,-1); update();}},
|
|
|
|
{type:"btn", font:ds, width:58, id:"OK", label:"OK", cb:callback}
|
|
|
|
]}
|
|
|
|
]}
|
|
|
|
], lazy:true});
|
|
|
|
g.clear();
|
|
|
|
numPad.render();
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
|
|
|
|
function removeCard() {
|
|
|
|
var menu = {
|
2023-10-12 12:30:30 +00:00
|
|
|
"" : {title : "Select WP"},
|
2023-03-14 22:08:30 +00:00
|
|
|
"< Back" : mainMenu
|
|
|
|
};
|
2023-10-12 12:30:30 +00:00
|
|
|
if (Object.keys(wp).length==0) Object.assign(menu, {"No WPs":""});
|
2023-06-26 19:26:42 +00:00
|
|
|
else {
|
|
|
|
wp.forEach((val, card) => {
|
|
|
|
const name = wp[card].name;
|
|
|
|
menu[name]=()=>{
|
2023-11-16 20:39:37 +00:00
|
|
|
E.showPrompt(name,{
|
|
|
|
title:"Delete",
|
|
|
|
}).then(function(v) {
|
|
|
|
if (v) {
|
|
|
|
wp.splice(card, 1);
|
|
|
|
writeWP();
|
|
|
|
mainMenu();
|
|
|
|
} else {
|
|
|
|
mainMenu();
|
|
|
|
}
|
|
|
|
});
|
2023-06-26 19:26:42 +00:00
|
|
|
};
|
|
|
|
});
|
2023-03-14 22:08:30 +00:00
|
|
|
}
|
|
|
|
E.showMenu(menu);
|
|
|
|
}
|
|
|
|
|
|
|
|
function ask01(t, cb) {
|
2023-10-12 12:30:30 +00:00
|
|
|
var la = new Layout (
|
2023-03-14 22:08:30 +00:00
|
|
|
{type:"v", c: [
|
2023-10-12 12:30:30 +00:00
|
|
|
{type:"txt", font:"15%", pad:1, fillx:1, filly:1, label:"Select"},
|
2023-03-14 22:08:30 +00:00
|
|
|
{type:"btn", font:"15%", pad:1, fillx:1, filly:1, label: t[0], cb:l=>{ cb(1); }},
|
|
|
|
{type:"btn", font:"15%", pad:1, fillx:1, filly:1, label: t[1], cb:l=>{ cb(-1); }},
|
|
|
|
], lazy:true});
|
|
|
|
g.clear();
|
2023-10-12 12:30:30 +00:00
|
|
|
la.render();
|
2023-03-14 22:08:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function askCoordinate(t1, t2, callback) {
|
|
|
|
let sign = 1;
|
|
|
|
ask01(t1, function(sign) {
|
|
|
|
switch (mode) {
|
|
|
|
case 0: s = "DDD.dddd"; break;
|
|
|
|
case 1: s = "DDD MM.mmm"; break;
|
|
|
|
case 2: s = "DDD MM'ss"+'"'; break;
|
|
|
|
}
|
|
|
|
showNumpad(s, t2, function() {
|
|
|
|
switch (mode) {
|
|
|
|
case 0:
|
|
|
|
res = parseFloat(key);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
d = parseInt(key.substr(0, 3));
|
|
|
|
m = parseFloat(key.substr(3,99));
|
|
|
|
res = d + m/60.0;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
d = parseInt(key.substr(0, 3));
|
|
|
|
m = parseInt(key.substr(4, 2));
|
|
|
|
s = parseInt(key.substr(7, 2));
|
|
|
|
res = d + m/60.0 + s/3600.0;
|
|
|
|
}
|
|
|
|
res = sign * res;
|
|
|
|
print("Coordinate", res);
|
|
|
|
callback(res);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function askPosition(callback) {
|
|
|
|
let full = "";
|
|
|
|
askCoordinate("NS", "0", function(lat) {
|
|
|
|
askCoordinate("EW", "", function(lon) {
|
|
|
|
callback(lat, lon);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-10-12 12:30:30 +00:00
|
|
|
function createWP(lat, lon, name) {
|
2023-11-03 20:56:10 +00:00
|
|
|
let n = {};
|
|
|
|
n["name"] = name;
|
|
|
|
n["lat"] = lat;
|
|
|
|
n["lon"] = lon;
|
|
|
|
wp.push(n);
|
|
|
|
print("add -- waypoints", wp);
|
|
|
|
writeWP();
|
2023-10-12 12:30:30 +00:00
|
|
|
}
|
|
|
|
|
2023-11-01 10:36:59 +00:00
|
|
|
function addCardName(name) {
|
2023-11-03 20:56:10 +00:00
|
|
|
g.clear();
|
|
|
|
askPosition(function(lat, lon) {
|
|
|
|
print("position -- ", lat, lon);
|
|
|
|
createWP(lat, lon, result);
|
|
|
|
mainMenu();
|
|
|
|
});
|
2023-11-01 10:36:59 +00:00
|
|
|
}
|
|
|
|
|
2023-03-14 22:08:30 +00:00
|
|
|
function addCard() {
|
2023-11-01 10:36:59 +00:00
|
|
|
require("textinput").input({text:"wp"}).then(key => {
|
2023-03-14 22:08:30 +00:00
|
|
|
result = key;
|
|
|
|
if (wp[result]!=undefined) {
|
|
|
|
E.showMenu();
|
|
|
|
var alreadyExists = new Layout (
|
|
|
|
{type:"v", c: [
|
|
|
|
{type:"txt", font:Math.min(15,100/result.length)+"%", pad:1, fillx:1, filly:1, label:result},
|
|
|
|
{type:"txt", font:"12%", pad:1, fillx:1, filly:1, label:"already exists."},
|
|
|
|
{type:"h", c: [
|
2023-11-01 10:36:59 +00:00
|
|
|
{type:"btn", font:"10%", pad:1, fillx:1, filly:1, label: "REPLACE", cb:l=>{addCardName(result);}},
|
2023-03-14 22:08:30 +00:00
|
|
|
{type:"btn", font:"10%", pad:1, fillx:1, filly:1, label: "CANCEL", cb:l=>{mainMenu();}}
|
|
|
|
]}
|
|
|
|
], lazy:true});
|
|
|
|
g.clear();
|
|
|
|
alreadyExists.render();
|
2023-11-01 10:36:59 +00:00
|
|
|
}
|
|
|
|
addCardName(result);
|
2023-03-14 22:08:30 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
g.reset();
|
|
|
|
mainMenu();
|