forked from FOSS/BangleApps
206 lines
5.2 KiB
JavaScript
206 lines
5.2 KiB
JavaScript
const Yoff = 80;
|
|
var pal2color = new Uint16Array([g.theme.bg,g.theme.fg,g.theme.fg2,0xC618],0,2);
|
|
var buf = Graphics.createArrayBuffer(240,60,2,{msb:true});
|
|
Bangle.setLCDTimeout(30);
|
|
|
|
function flip(b,y) {
|
|
g.drawImage({width:240,height:60,bpp:2,buffer:b.buffer, palette:pal2color},0,y);
|
|
b.clear();
|
|
}
|
|
|
|
const labels = ["N","NE","E","SE","S","SW","W","NW"];
|
|
var brg=null;
|
|
|
|
function drawCompass(course) {
|
|
"ram"
|
|
buf.setColor(1);
|
|
buf.setFont("Vector",24);
|
|
var start = course-90;
|
|
if (start<0) start+=360;
|
|
buf.fillRect(28,45,212,49);
|
|
var xpos = 30;
|
|
var frag = 15 - start%15;
|
|
if (frag<15) xpos+=frag; else frag = 0;
|
|
for (var i=frag;i<=180-frag;i+=15){
|
|
var res = start + i;
|
|
if (res%90==0) {
|
|
buf.drawString(labels[Math.floor(res/45)%8],xpos-8,0);
|
|
buf.fillRect(xpos-2,25,xpos+2,45);
|
|
} else if (res%45==0) {
|
|
buf.drawString(labels[Math.floor(res/45)%8],xpos-12,0);
|
|
buf.fillRect(xpos-2,30,xpos+2,45);
|
|
} else if (res%15==0) {
|
|
buf.fillRect(xpos,35,xpos+1,45);
|
|
}
|
|
xpos+=15;
|
|
}
|
|
if (brg) {
|
|
var bpos = brg - course;
|
|
if (bpos>180) bpos -=360;
|
|
if (bpos<-180) bpos +=360;
|
|
bpos+=120;
|
|
if (bpos<30) bpos = 14;
|
|
if (bpos>210) bpos = 226;
|
|
buf.setColor(2);
|
|
buf.fillCircle(bpos,40,8);
|
|
}
|
|
flip(buf,Yoff);
|
|
}
|
|
|
|
var heading = 0;
|
|
function newHeading(m,h){
|
|
var s = Math.abs(m - h);
|
|
var delta = (m>h)?1:-1;
|
|
if (s>=180){s=360-s; delta = -delta;}
|
|
if (s<2) return h;
|
|
var hd = h + delta*(1 + Math.round(s/5));
|
|
if (hd<0) hd+=360;
|
|
if (hd>360)hd-= 360;
|
|
return hd;
|
|
}
|
|
|
|
var candraw = false;
|
|
var CALIBDATA = require("Storage").readJSON("magnav.json",1)||null;
|
|
|
|
function tiltfixread(O,S){
|
|
"ram"
|
|
var m = Bangle.getCompass();
|
|
var g = Bangle.getAccel();
|
|
m.dx =(m.x-O.x)*S.x; m.dy=(m.y-O.y)*S.y; m.dz=(m.z-O.z)*S.z;
|
|
var d = Math.atan2(-m.dx,m.dy)*180/Math.PI;
|
|
if (d<0) d+=360;
|
|
var phi = Math.atan(-g.x/-g.z);
|
|
var cosphi = Math.cos(phi), sinphi = Math.sin(phi);
|
|
var theta = Math.atan(-g.y/(-g.x*sinphi-g.z*cosphi));
|
|
var costheta = Math.cos(theta), sintheta = Math.sin(theta);
|
|
var xh = m.dy*costheta + m.dx*sinphi*sintheta + m.dz*cosphi*sintheta;
|
|
var yh = m.dz*sinphi - m.dx*cosphi;
|
|
var psi = Math.atan2(yh,xh)*180/Math.PI;
|
|
if (psi<0) psi+=360;
|
|
return psi;
|
|
}
|
|
|
|
// Note actual mag is 360-m, error in firmware
|
|
function reading() {
|
|
var d = tiltfixread(CALIBDATA.offset,CALIBDATA.scale);
|
|
heading = newHeading(d,heading);
|
|
drawCompass(heading);
|
|
buf.setColor(1);
|
|
buf.setFont("6x8",2);
|
|
buf.setFontAlign(-1,-1);
|
|
buf.drawString("o",170,0);
|
|
buf.setFont("Vector",54);
|
|
var course = Math.round(heading);
|
|
var cs = course.toString();
|
|
cs = course<10?"00"+cs : course<100 ?"0"+cs : cs;
|
|
buf.drawString(cs,70,10);
|
|
flip(buf,Yoff+80);
|
|
}
|
|
|
|
function calibrate(){
|
|
var max={x:-32000, y:-32000, z:-32000},
|
|
min={x:32000, y:32000, z:32000};
|
|
var ref = setInterval(()=>{
|
|
var m = Bangle.getCompass();
|
|
max.x = m.x>max.x?m.x:max.x;
|
|
max.y = m.y>max.y?m.y:max.y;
|
|
max.z = m.z>max.z?m.z:max.z;
|
|
min.x = m.x<min.x?m.x:min.x;
|
|
min.y = m.y<min.y?m.y:min.y;
|
|
min.z = m.z<min.z?m.z:min.z;
|
|
}, 100);
|
|
return new Promise((resolve) => {
|
|
setTimeout(()=>{
|
|
if(ref) clearInterval(ref);
|
|
var offset = {x:(max.x+min.x)/2,y:(max.y+min.y)/2,z:(max.z+min.z)/2};
|
|
var delta = {x:(max.x-min.x)/2,y:(max.y-min.y)/2,z:(max.z-min.z)/2};
|
|
var avg = (delta.x+delta.y+delta.z)/3;
|
|
var scale = {x:avg/delta.x, y:avg/delta.y, z:avg/delta.z};
|
|
resolve({offset:offset,scale:scale});
|
|
},20000);
|
|
});
|
|
}
|
|
|
|
var calibrating=false;
|
|
function docalibrate(first){
|
|
calibrating=true;
|
|
const title = "Calibrate";
|
|
const msg = "takes 20 seconds";
|
|
function restart() {
|
|
calibrating=false;
|
|
setButtons();
|
|
startdraw();
|
|
}
|
|
function action(b){
|
|
if (b) {
|
|
buf.setColor(1);
|
|
buf.setFont("Vector",24);
|
|
buf.setFontAlign(0,-1);
|
|
buf.drawString("Fig 8s to",120,0);
|
|
buf.drawString("Calibrate",120,26);
|
|
flip(buf,Yoff);
|
|
calibrate().then((r)=>{
|
|
CALIBDATA=r;
|
|
require("Storage").write("magnav.json",r);
|
|
restart()
|
|
});
|
|
} else {
|
|
restart()
|
|
}
|
|
}
|
|
if (first===undefined) first=false;
|
|
stopdraw();
|
|
if (first)
|
|
E.showAlert(msg,title).then(action.bind(null,true));
|
|
else
|
|
E.showPrompt(msg,{title:title,buttons:{"Start":true,"Cancel":false}}).then(action);
|
|
}
|
|
|
|
var intervalRef;
|
|
|
|
function startdraw(){
|
|
g.clear();
|
|
g.setColor(1,0.5,0.5);
|
|
g.fillPoly([120,Yoff+60,110,Yoff+80,130,Yoff+80]);
|
|
g.setColor(1,1,1);
|
|
Bangle.drawWidgets();
|
|
candraw = true;
|
|
intervalRef = setInterval(reading,200);
|
|
}
|
|
|
|
function stopdraw() {
|
|
candraw=false;
|
|
if(intervalRef) {clearInterval(intervalRef);}
|
|
}
|
|
|
|
function setButtons(){
|
|
function actions(v){
|
|
if (!v) docalibrate(false);
|
|
else if (v==1) brg=null;
|
|
else brg=heading;
|
|
}
|
|
Bangle.setUI("updown",actions);
|
|
}
|
|
|
|
Bangle.on('lcdPower',function(on) {
|
|
if (on) {
|
|
if (!calibrating) startdraw();
|
|
} else {
|
|
stopdraw();
|
|
}
|
|
});
|
|
|
|
Bangle.on('kill',()=>{Bangle.setCompassPower(0);});
|
|
|
|
Bangle.loadWidgets();
|
|
Bangle.setCompassPower(1);
|
|
if (!CALIBDATA)
|
|
docalibrate(true);
|
|
else {
|
|
startdraw();
|
|
setButtons();
|
|
}
|
|
|
|
|
|
|