BangleApps/apps/openstmap/app.js

282 lines
8.2 KiB
JavaScript
Raw Normal View History

var m = require("openstmap");
var HASWIDGETS = true;
var R;
var fix = {};
var mapVisible = false;
var hasScrolled = false;
var settings = require("Storage").readJSON("openstmap.json",1)||{};
2023-04-25 09:09:07 +00:00
var plotTrack;
let checkMapPos = false; // Do we need to check the if the coordinates we have are valid
var startDrag = 0;
2023-07-22 18:06:53 +00:00
if (Bangle.setLCDOverlay) {
2023-07-22 18:09:21 +00:00
// Icon for current location+direction: https://icons8.com/icon/11932/gps 24x24, 1 Bit + transparency + inverted
2023-07-22 18:06:53 +00:00
var imgLoc = require("heatshrink").decompress(atob("jEYwINLAQk8AQl+AQn/AQcB/+AAQUD//AAQUH//gAQUP//wAQUf//4j8AvA9IA=="));
// overlay buffer for current location, a bit bigger then image so we can rotate
const ovSize = Math.ceil(Math.sqrt(imgLoc[0]*imgLoc[0]+imgLoc[1]*imgLoc[1]));
var ovLoc = Graphics.createArrayBuffer(ovSize,ovSize,imgLoc[2] & 0x7f,{msb:true});
}
if (settings.lat !== undefined && settings.lon !== undefined && settings.scale !== undefined) {
// restore last view
m.lat = settings.lat;
m.lon = settings.lon;
m.scale = settings.scale;
checkMapPos = true;
}
if (settings.dirSrc === undefined) {
settings.dirSrc = 1; // Default=GPS
}
// Redraw the whole page
function redraw() {
// ensure we do cancel track drawing
if (plotTrack && plotTrack.stop)
plotTrack.stop();
// set clip rect so we don't overwrite widgets
g.setClipRect(R.x,R.y,R.x2,R.y2);
const count = m.draw();
if (checkMapPos && count === 0) {
// no map at these coordinates, lets try again with first map
m.lat = m.map.lat;
m.lon = m.map.lon;
m.scale = m.map.scale;
m.draw();
}
checkMapPos = false;
drawPOI();
drawMarker();
drawLocation();
// if track drawing is enabled...
if (settings.drawTrack) {
if (HASWIDGETS && WIDGETS["gpsrec"] && WIDGETS["gpsrec"].plotTrack) {
g.setColor("#f00").flip(); // force immediate draw on double-buffered screens - track will update later
WIDGETS["gpsrec"].plotTrack(m);
}
if (HASWIDGETS && WIDGETS["recorder"] && WIDGETS["recorder"].plotTrack) {
g.setColor("#f00").flip(); // force immediate draw on double-buffered screens - track will update later
2023-04-25 09:09:07 +00:00
plotTrack = WIDGETS["recorder"].plotTrack(m, { async : true, callback : function() {
plotTrack = undefined;
}});
}
}
g.setClipRect(0,0,g.getWidth()-1,g.getHeight()-1);
}
// Draw the POIs
function drawPOI() {
2024-01-27 09:15:31 +00:00
let waypoints;
try {
2024-01-27 09:15:31 +00:00
waypoints = require("waypoints").load();
} catch (ex) {
// Waypoints module not available.
return;
}
g.setFont("Vector", 18);
waypoints.forEach((wp, idx) => {
2024-01-27 09:15:31 +00:00
if (wp.lat === undefined || wp.lon === undefined) return;
var p = m.latLonToXY(wp.lat, wp.lon);
var sz = 2;
g.setColor(0,0,0);
g.fillRect(p.x-sz, p.y-sz, p.x+sz, p.y+sz);
g.setColor(0,0,0);
g.drawString(wp.name, p.x, p.y);
//print(wp.name);
2024-01-27 09:15:31 +00:00
});
}
function isInside(rect, e, w, h) {
return e.x-w/2>=rect.x && e.x+w/2<rect.x+rect.w
&& e.y-h/2>=rect.y && e.y+h/2<=rect.y+rect.h;
}
// Draw the location & direction marker for where we are
function drawMarker() {
if (!fix.fix || !settings.drawMarker) return;
var p = m.latLonToXY(fix.lat, fix.lon);
if (isInside(R, p, 4, 4)) { // avoid drawing over widget area
g.setColor(1,0,0);
g.fillRect(p.x-2, p.y-2, p.x+2, p.y+2);
}
}
// Draw current location+direction with LCD Overlay (Bangle.js 2 only)
function drawLocation() {
if (!Bangle.setLCDOverlay) {
return; // Overlay not supported
}
if (!fix.fix || !mapVisible || settings.dirSrc === 0) {
if (this.hasOverlay) {
Bangle.setLCDOverlay(); // clear if map is not visible or no fix
this.hasOverlay = false;
}
return;
}
var p = m.latLonToXY(fix.lat, fix.lon);
ovLoc.clear();
if (isInside(R, p, ovLoc.getWidth(), ovLoc.getHeight())) { // avoid drawing over widget area
const angle = settings.dirSrc === 1 ? fix.course : Bangle.getCompass().heading;
if (!isNaN(angle)) {
ovLoc.drawImage(imgLoc, ovLoc.getWidth()/2, ovLoc.getHeight()/2, {rotate: angle*Math.PI/180});
}
}
Bangle.setLCDOverlay({width:ovLoc.getWidth(), height:ovLoc.getHeight(),
2023-07-22 18:06:53 +00:00
bpp:ovLoc.getBPP(), transparent:0,
palette:new Uint16Array([0, g.toColor("#00F")]),
buffer:ovLoc.buffer
}, p.x-ovLoc.getWidth()/2, p.y-ovLoc.getHeight()/2);
this.hasOverlay = true;
}
Bangle.on('GPS',function(f) {
fix=f;
if (HASWIDGETS && WIDGETS["sats"]) WIDGETS["sats"].draw(WIDGETS["sats"]);
if (mapVisible) {
drawMarker();
drawLocation();
}
});
Bangle.setGPSPower(1, "app");
Bangle.setCompassPower(settings.dirSrc === 2, "openstmap");
if (HASWIDGETS) {
Bangle.loadWidgets();
2023-07-04 19:58:04 +00:00
if (!WIDGETS["gps"]) { // one GPS Widget is enough
WIDGETS["sats"] = { area:"tl", width:24, draw:w=>{
var sats = 0|fix.satellites;
g.reset().clearRect(w.x,w.y,w.x+23,w.y+23). drawImage(atob("EhPCAP//vmU0Cv++BUAAAANVQAAANVUAAANVVSoA/VVWqA/VqqqA/1VVoA/9VVkAP/VVlAP/1VVAP/9VVQD//WVUA//2VUA//9VUAX//1UBV///0BVX//ABVVAAABVVAAAA="), w.x,w.y+5); // 18x19
if (fix.fix) g.setFont("6x8").setFontAlign(1,-1).drawString(sats,w.x+23,w.y);
else g.setFont("4x6:2").setColor("#f00").setFontAlign(1,-1).drawString("X",w.x+23,w.y);
}};
}
Bangle.drawWidgets();
}
R = Bangle.appRect;
function writeSettings() {
settings.lat = m.lat;
settings.lon = m.lon;
settings.scale = m.scale;
require("Storage").writeJSON("openstmap.json",settings);
}
function showMenu() {
if (plotTrack && plotTrack.stop)
plotTrack.stop();
mapVisible = false;
drawLocation();
var menu = {
"":{title:/*LANG*/"Map"},
"< Back": ()=> showMap(),
};
// If we have a GPS fix, add a menu item to center it
if (fix.fix) menu[/*LANG*/"Center GPS"]=() =>{
m.lat = fix.lat;
m.lon = fix.lon;
showMap();
};
menu = Object.assign(menu, {
/*LANG*/"Zoom In": () =>{
m.scale /= 2;
showMap();
},
/*LANG*/"Zoom Out": () =>{
m.scale *= 2;
showMap();
},
/*LANG*/"Draw Track": {
value : !!settings.drawTrack,
onchange : v => { settings.drawTrack=v; writeSettings(); }
},
/*LANG*/"Draw cont. position": {
value : !!settings.drawMarker,
onchange : v => { settings.drawMarker=v; writeSettings(); }
},
});
if (Bangle.setLCDOverlay) {
menu[/*LANG*/"Direction source"] = {
value: settings.dirSrc,
min: 0, max: 2,
format: v => [/*LANG*/"None", /*LANG*/"GPS", /*LANG*/"Compass"][v],
onchange: v => {
settings.dirSrc = v;
Bangle.setCompassPower(settings.dirSrc === 2, "openstmap");
writeSettings();
}
};
menu[/*LANG*/"Reset compass"] = () => {
Bangle.resetCompass();
showMap();
};
}
menu[/*LANG*/"Center Map"] = () =>{
m.lat = m.map.lat;
m.lon = m.map.lon;
m.scale = m.map.scale;
showMap();
};
// If we have the recorder widget, add a menu item to start/stop recording
if (WIDGETS.recorder) {
menu[/*LANG*/"Record"] = {
value : WIDGETS.recorder.isRecording(),
onchange : isOn => {
E.showMessage(/*LANG*/"Please Wait...");
WIDGETS.recorder.setRecording(isOn).then(showMap);
}
};
}
menu[/*LANG*/"Exit"] = () => load();
E.showMenu(menu);
}
function showMap() {
mapVisible = true;
g.reset().clearRect(R);
redraw();
Bangle.setUI({mode:"custom",drag:e=>{
2023-04-25 09:09:07 +00:00
if (plotTrack && plotTrack.stop) plotTrack.stop();
if (e.b) {
if (!startDrag)
startDrag = getTime();
g.setClipRect(R.x,R.y,R.x2,R.y2);
g.scroll(e.dx,e.dy);
m.scroll(e.dx,e.dy);
g.setClipRect(0,0,g.getWidth()-1,g.getHeight()-1);
hasScrolled = true;
drawLocation();
} else if (hasScrolled) {
2024-03-13 10:51:40 +00:00
const delta = getTime() - startDrag;
startDrag = 0;
hasScrolled = false;
if (delta < 0.2) {
if (e.y > g.getHeight() / 2) {
if (e.x < g.getWidth() / 2) {
m.scale /= 2;
} else {
m.scale *= 2;
}
}
g.reset().clearRect(R);
}
redraw();
}
}, btn: () => showMenu() });
}
if (m.maps.length === 0) {
E.showPrompt(/*LANG*/'Please upload a map first.', {buttons : {/*LANG*/"Ok":true}}).then(v => load());
} else {
showMap();
}
// Write settings on exit via button
setWatch(() => writeSettings(), BTN, { repeat: true, edge: "rising" });