BangleApps/apps/circlesclock/app.js

471 lines
13 KiB
JavaScript
Raw Normal View History

2021-12-16 07:06:53 +00:00
const locale = require("locale");
const heatshrink = require("heatshrink");
const storage = require("Storage");
2021-12-16 07:06:53 +00:00
const shoesIcon = heatshrink.decompress(atob("h0OwYJGgmAAgUBkgECgVJB4cSoAUDyEBkARDpADBhMAyQRBgVAkgmDhIUDAAuQAgY1DAAYA="));
const shoesIconGreen = heatshrink.decompress(atob("h0OwYJGhIEDgVIAgUEyQKDkmACgcggVACIeQAYMSgIRCgmApIbDiQUDAAkBkAFDGoYAD"));
const heartIcon = heatshrink.decompress(atob("h0OwYOLkmQhMkgACByVJgESpIFBpEEBAIFBCgIFCCgsABwcAgQOCAAMSpAwDyBNM"));
const powerIcon = heatshrink.decompress(atob("h0OwYQNsAED7AEDmwEDtu2AgUbtuABwXbBIUN23AAoYOCgEDFIgODABI"));
const powerIconGreen = heatshrink.decompress(atob("h0OwYQNkAEDpAEDiQEDkmSAgUJkmABwVJBIUEyVAAoYOCgEBFIgODABI"));
const powerIconRed = heatshrink.decompress(atob("h0OwYQNoAEDyAEDkgEDpIFDiVJBweSAgUJkmAAoYZDgQpEBwYAJA"));
2021-12-16 07:06:53 +00:00
const weatherCloudy = heatshrink.decompress(atob("iEQwYWTgP//+AAoMPAoPwAoN/AocfAgP//0AAgQAB/AFEABgdDAAMDDohMRA"));
const weatherSunny = heatshrink.decompress(atob("iEQwYLIg3AAgVgAQMMAo8Am3YAgUB23bAoUNAoIUBjYFCsOwBYoFDDpFgHYI1JI4gFGAAYA="));
const weatherPartlyCloudy = heatshrink.decompress(atob("iEQwYQNv0AjgGDn4EDh///gFChwREC4MfxwIBv0//+AC4X4j4FCv/AgfwgED/wIBuAaBBwgFDgP4gf/AAXABwIEBDQQAEA=="));
const weatherRainy = heatshrink.decompress(atob("iEQwYLIg/gAgUB///wAFBh/AgfwgED/wIBuEAj4OCv0AjgaCh/4AocAnAFBFIU4EAM//gRBEAIOBhw1C/AmDAosAC4JNIAAg"));
const weatherPartlyRainy = heatshrink.decompress(atob("h0OwYJGjkAnAFCj+AAgU//4FCuEA8EAg8ch/4gEB4////AAoIIBCIMD/wgCg4bBg/8BwMD+AgBh4ZBDQf/FIIABh4IBgAA=="));
const weatherSnowy = heatshrink.decompress(atob("iEQwYROn/8AocH8AECuAFBh0Agf+CIN/4EDx/4j/x4EAgIIBwAXBAogRFDoopFGoxBGABIA="));
const weatherFoggy = heatshrink.decompress(atob("iEQwYROn/8AgUB/EfwAFBh/AgfwgED/wIBuEABwd/4EcDQgFDgE4Fosf///8f//A/Lj/xCQIRNA="));
const weatherStormy = heatshrink.decompress(atob("iEQwYLIg/gAgUB///wAFBh/AgfwgED/wIBuEAj4OCv0AjgaCh/4AoX8gE4AoQpBnAdBF4IRBDQMH/kOHgY7DAo4AOA=="));
2021-12-16 07:06:53 +00:00
let settings;
2021-12-16 07:06:53 +00:00
function loadSettings() {
settings = storage.readJSON("circlesclock.json", 1) || {
'minHR': 40,
2021-12-16 07:06:53 +00:00
'maxHR': 200,
'stepGoal': 10000,
'stepDistanceGoal': 8000,
'stepLength': 0.8,
'batteryWarn': 30,
'showWidgets': false,
'weatherCircleData': 'humidity',
'circle1': 'hr',
'circle2': 'steps',
'circle3': 'battery'
2021-12-16 07:06:53 +00:00
};
// Load step goal from pedometer widget as fallback
if (settings.stepGoal == undefined) {
const d = require('Storage').readJSON("wpedom.json", 1) || {};
settings.stepGoal = d != undefined && d.settings != undefined ? d.settings.goal : 10000;
}
2021-12-16 07:06:53 +00:00
}
loadSettings();
const showWidgets = settings.showWidgets || false;
let hrtValue;
2021-12-16 07:06:53 +00:00
// layout values:
2021-12-17 21:36:34 +00:00
const colorFg = g.theme.dark ? '#fff' : '#000';
const colorBg = g.theme.dark ? '#000' : '#fff';
2021-12-16 07:06:53 +00:00
const colorGrey = '#808080';
const colorRed = '#ff0000';
const colorGreen = '#008000';
const colorBlue = '#0000ff';
const colorYellow = '#ffff00';
2022-01-06 10:08:46 +00:00
const widgetOffset = showWidgets ? 24 : 0;
const h = g.getHeight() - widgetOffset;
2021-12-16 07:06:53 +00:00
const w = g.getWidth();
const hOffset = 30 - widgetOffset;
2021-12-16 07:06:53 +00:00
const h1 = Math.round(1 * h / 5 - hOffset);
const h2 = Math.round(3 * h / 5 - hOffset);
const h3 = Math.round(8 * h / 8 - hOffset - 3); // circle y position
const circlePosX = [Math.round(w / 6), Math.round(3 * w / 6), Math.round(5 * w / 6)]; // cirle x positions
const radiusOuter = 25;
const radiusInner = 20;
const circleFont = "Vector:15";
const circleFontBig = "Vector:16";
2021-12-16 07:06:53 +00:00
function draw() {
2022-01-05 18:38:18 +00:00
g.clear(true);
2022-01-05 18:50:07 +00:00
if (!showWidgets) {
/*
* we are not drawing the widgets as we are taking over the whole screen
* so we will blank out the draw() functions of each widget and change the
* area to the top bar doesn't get cleared.
*/
2022-01-05 18:50:07 +00:00
if (WIDGETS && typeof WIDGETS === "object") {
for (let wd of WIDGETS) {
wd.draw = () => {};
wd.area = "";
}
}
} else {
Bangle.drawWidgets();
}
2021-12-16 07:06:53 +00:00
g.setColor(colorBg);
2022-01-05 18:38:18 +00:00
g.fillRect(0, widgetOffset, w, h);
2021-12-16 07:06:53 +00:00
// time
g.setFont("Vector:50");
g.setFontAlign(0, -1);
2021-12-16 07:06:53 +00:00
g.setColor(colorFg);
g.drawString(locale.time(new Date(), 1), w / 2, h1 + 8);
2021-12-16 07:06:53 +00:00
// date & dow
g.setFont("Vector:21");
2021-12-16 07:06:53 +00:00
g.setFontAlign(-1, 0);
g.drawString(locale.date(new Date()), w > 180 ? 2 * w / 10 : w / 10, h2);
g.drawString(locale.dow(new Date()), w > 180 ? 2 * w / 10 : w / 10, h2 + 22);
2021-12-16 07:06:53 +00:00
drawCircle(1);
drawCircle(2);
drawCircle(3);
}
2021-12-16 07:06:53 +00:00
const defaultCircleTypes = ["steps", "hr", "battery"];
2021-12-16 07:06:53 +00:00
function drawCircle(index) {
let type = settings['circle' + index];
if (!type) type = defaultCircleTypes[index - 1];
const w = getCirclePosition(type);
switch (type) {
case "steps":
drawSteps(w);
break;
case "stepsDist":
drawStepsDistance(w);
break;
case "hr":
drawHeartRate(w);
break;
case "battery":
drawBattery(w);
break;
case "weather":
drawWeather(w);
break;
2022-01-14 13:46:59 +00:00
case "empty":
// we do nothing here
return;
}
}
2022-01-14 14:19:00 +00:00
// serves as cache for quicker lookup of circle positions
let circlePositionsCache = [];
/*
* Looks in the following order if a circle with the given type is somewhere visible/configured
* 1. circlePositionsCache
* 2. settings
* 3. defaultCircleTypes
*
* In case 2 and 3 the circlePositionsCache will be updated
*/
function getCirclePosition(type) {
2022-01-14 14:19:00 +00:00
if (circlePositionsCache[type] >= 0) {
return circlePosX[circlePositionsCache[type]];
}
for (let i = 1; i <= 3; i++) {
const setting = settings['circle' + i];
2022-01-14 14:19:00 +00:00
if (setting == type) {
circlePositionsCache[type] = i - 1;
return circlePosX[i - 1];
}
}
for (let i = 0; i < defaultCircleTypes.length; i++) {
2022-01-14 14:19:00 +00:00
if (type == defaultCircleTypes[i] && (!settings || settings['circle' + (i + 1)] == undefined)) {
circlePositionsCache[type] = i;
return circlePosX[i];
}
}
return undefined;
2021-12-16 07:06:53 +00:00
}
function isCircleEnabled(type) {
return getCirclePosition(type) != undefined;
}
2021-12-16 07:06:53 +00:00
function drawSteps(w) {
if (!w) w = getCirclePosition("steps");
2021-12-16 07:06:53 +00:00
const steps = getSteps();
drawCircleBackground(w);
2021-12-16 07:06:53 +00:00
2021-12-17 21:36:34 +00:00
const stepGoal = settings.stepGoal || 10000;
2021-12-16 07:06:53 +00:00
if (stepGoal > 0) {
let percent = steps / stepGoal;
if (stepGoal < steps) percent = 1;
drawGauge(w, h3, percent, colorBlue);
2021-12-16 07:06:53 +00:00
}
drawInnerCircleAndTriangle(w);
2021-12-16 07:06:53 +00:00
writeCircleText(w, shortValue(steps));
2021-12-16 07:06:53 +00:00
g.drawImage(shoesIcon, w - 6, h3 + radiusOuter - 6);
2021-12-16 07:06:53 +00:00
}
function drawStepsDistance(w) {
if (!w) w = getCirclePosition("steps");
const steps = getSteps();
const stepDistance = settings.stepLength || 0.8;
2022-01-06 12:01:36 +00:00
const stepsDistance = Math.round(steps * stepDistance);
drawCircleBackground(w);
const stepDistanceGoal = settings.stepDistanceGoal || 8000;
if (stepDistanceGoal > 0) {
let percent = stepsDistance / stepDistanceGoal;
if (stepDistanceGoal < stepsDistance) percent = 1;
drawGauge(w, h3, percent, colorGreen);
}
drawInnerCircleAndTriangle(w);
writeCircleText(w, shortValue(stepsDistance));
g.drawImage(shoesIconGreen, w - 6, h3 + radiusOuter - 6);
}
function drawHeartRate(w) {
if (!w) w = getCirclePosition("hr");
drawCircleBackground(w);
2021-12-16 07:06:53 +00:00
2022-01-14 13:46:59 +00:00
if (hrtValue != undefined) {
const minHR = settings.minHR || 40;
const maxHR = settings.maxHR || 200;
const percent = (hrtValue - minHR) / (maxHR - minHR);
drawGauge(w, h3, percent, colorRed);
2021-12-16 07:06:53 +00:00
}
drawInnerCircleAndTriangle(w);
2021-12-16 07:06:53 +00:00
writeCircleText(w, hrtValue != undefined ? hrtValue : "-");
2021-12-16 07:06:53 +00:00
g.drawImage(heartIcon, w - 6, h3 + radiusOuter - 6);
2021-12-16 07:06:53 +00:00
}
function drawBattery(w) {
if (!w) w = getCirclePosition("battery");
2021-12-16 07:06:53 +00:00
const battery = E.getBattery();
drawCircleBackground(w);
2021-12-16 07:06:53 +00:00
if (battery > 0) {
const percent = battery / 100;
drawGauge(w, h3, percent, colorYellow);
2021-12-16 07:06:53 +00:00
}
drawInnerCircleAndTriangle(w);
let icon = powerIcon;
let color = colorFg;
if (Bangle.isCharging()) {
color = colorGreen;
icon = powerIconGreen;
} else {
if (settings.batteryWarn != undefined && battery <= settings.batteryWarn) {
color = colorRed;
icon = powerIconRed;
}
}
writeCircleText(w, battery + '%');
2021-12-16 07:06:53 +00:00
g.drawImage(icon, w - 6, h3 + radiusOuter - 6);
2021-12-16 07:06:53 +00:00
}
function drawWeather(w) {
if (!w) w = getCirclePosition("weather");
const weather = getWeather();
const tempString = weather ? locale.temp(weather.temp - 273.15) : undefined;
const code = weather ? weather.code : -1;
drawCircleBackground(w);
const data = settings.weatherCircleData || "humidity";
switch (data) {
case "humidity":
const humidity = weather ? weather.hum : undefined;
if (humidity >= 0) {
drawGauge(w, h3, humidity / 100, colorYellow);
}
break;
case "wind":
const wind = weather ? weather.wind : undefined;
if (wind >= 0) {
// wind goes from 0 to 12 (see https://en.wikipedia.org/wiki/Beaufort_scale)
drawGauge(w, h3, wind / 12, colorYellow);
}
break;
case "empty":
break;
2022-01-11 12:52:19 +00:00
}
drawInnerCircleAndTriangle(w);
writeCircleText(w, tempString ? tempString : "?");
if (code > 0) {
const icon = getWeatherIconByCode(code);
if (icon) g.drawImage(icon, w - 6, h3 + radiusOuter - 10);
}
}
/*
* Choose weather icon to display based on weather conditition code
* https://openweathermap.org/weather-conditions#Weather-Condition-Codes-2
*/
function getWeatherIconByCode(code) {
const codeGroup = Math.round(code / 100);
switch (codeGroup) {
case 2:
return weatherStormy;
case 3:
return weatherCloudy;
case 5:
switch (code) {
case 511:
return weatherSnowy;
case 520:
return weatherPartlyRainy;
case 521:
return weatherPartlyRainy;
case 522:
return weatherPartlyRainy;
case 531:
return weatherPartlyRainy;
default:
return weatherRainy;
}
break;
case 6:
return weatherSnowy;
case 7:
return weatherFoggy;
case 8:
switch (code) {
case 800:
return weatherSunny;
case 801:
return weatherPartlyCloudy;
case 802:
return weatherPartlyCloudy;
default:
return weatherCloudy;
}
break;
default:
return undefined;
}
return undefined;
}
/*
* Draws the background and the grey circle
*/
function drawCircleBackground(w) {
// Draw rectangle background:
g.setColor(colorBg);
g.fillRect(w - radiusOuter - 3, h3 - radiusOuter - 3, w + radiusOuter + 3, h3 + radiusOuter + 3);
// Draw grey background circle:
g.setColor(colorGrey);
g.fillCircle(w, h3, radiusOuter);
}
function drawInnerCircleAndTriangle(w) {
// Draw inner circle
g.setColor(colorBg);
g.fillCircle(w, h3, radiusInner);
// Draw triangle which covers the bottom of the circle
g.fillPoly([w, h3, w - 15, h3 + radiusOuter + 5, w + 15, h3 + radiusOuter + 5]);
}
2021-12-16 07:06:53 +00:00
function radians(a) {
return a * Math.PI / 180;
}
/*
* This draws the actual gauge consisting out of lots of little filled circles
*/
2021-12-16 07:06:53 +00:00
function drawGauge(cx, cy, percent, color) {
const offset = 15;
const end = 345;
const radius = radiusInner + 3;
const size = radiusOuter - radiusInner - 2;
2021-12-16 07:06:53 +00:00
if (percent <= 0) return;
2021-12-16 07:06:53 +00:00
if (percent > 1) percent = 1;
const startRotation = -offset;
const endRotation = startRotation - ((end - offset) * percent);
2021-12-16 07:06:53 +00:00
g.setColor(color);
for (let i = startRotation; i > endRotation - size; i -= size) {
x = cx + radius * Math.sin(radians(i));
y = cy + radius * Math.cos(radians(i));
2021-12-17 21:36:34 +00:00
g.fillCircle(x, y, size);
2021-12-16 07:06:53 +00:00
}
}
function writeCircleText(w, content) {
if (content == undefined) return;
g.setFont(content.length < 4 ? circleFontBig : circleFont);
g.setFontAlign(0, 0);
g.setColor(colorFg);
g.drawString(content, w, h3);
}
2021-12-16 07:06:53 +00:00
function shortValue(v) {
if (isNaN(v)) return '-';
if (v <= 999) return v;
if (v >= 1000 && v < 10000) {
v = Math.floor(v / 100) * 100;
return (v / 1000).toFixed(1).replace(/\.0$/, '') + 'k';
}
if (v >= 10000) {
v = Math.floor(v / 1000) * 1000;
return (v / 1000).toFixed(1).replace(/\.0$/, '') + 'k';
}
}
function getSteps() {
2022-01-05 18:38:18 +00:00
if (WIDGETS && WIDGETS.wpedom !== undefined) {
2021-12-16 07:06:53 +00:00
return WIDGETS.wpedom.getSteps();
}
return 0;
}
function getWeather() {
const jsonWeather = storage.readJSON('weather.json');
return jsonWeather && jsonWeather.weather ? jsonWeather.weather : undefined;
}
function enableHRMSensor() {
Bangle.setHRMPower(1, "circleclock");
if (hrtValue == undefined) {
hrtValue = '...';
drawHeartRate();
}
}
2021-12-16 07:06:53 +00:00
Bangle.on('lock', function(isLocked) {
if (!isLocked) {
if (isCircleEnabled("hr")) {
enableHRMSensor();
2021-12-17 21:36:34 +00:00
}
2022-01-05 18:50:07 +00:00
draw();
2021-12-16 07:06:53 +00:00
} else {
Bangle.setHRMPower(0, "circleclock");
2021-12-16 07:06:53 +00:00
}
});
Bangle.on('HRM', function(hrm) {
if (isCircleEnabled("hr")) {
hrtValue = hrm.bpm;
if (Bangle.isLCDOn())
drawHeartRate();
}
});
Bangle.setUI("clock");
Bangle.loadWidgets();
draw();
setInterval(draw, 60000);
Bangle.on('charging', function(charging) {
if (isCircleEnabled("battery")) drawBattery();
});
if (isCircleEnabled("hr")) {
enableHRMSensor();
}