|
|
|
@ -1,10 +1,28 @@
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
* Includes
|
|
|
|
|
*/
|
|
|
|
|
const locale = require('locale');
|
|
|
|
|
const ts = 150;
|
|
|
|
|
const clock_info = require("clock_info");
|
|
|
|
|
const storage = require('Storage');
|
|
|
|
|
const locale = require('locale');
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
* Settings
|
|
|
|
|
*/
|
|
|
|
|
const SETTINGS_FILE = "bwclk.setting.json";
|
|
|
|
|
let settings = {
|
|
|
|
|
menuPosX: 0,
|
|
|
|
|
menuPosY: 0,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let saved_settings = storage.readJSON(SETTINGS_FILE, 1) || settings;
|
|
|
|
|
for (const key in saved_settings) {
|
|
|
|
|
settings[key] = saved_settings[key]
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
* Assets
|
|
|
|
|
*/
|
|
|
|
|
Graphics.prototype.setFontUbuntuMono = function(scale) {
|
|
|
|
@ -16,57 +34,162 @@ const ts = 150;
|
|
|
|
|
28+(scale<<8)+(1<<16)
|
|
|
|
|
);
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
* Menu
|
|
|
|
|
*/
|
|
|
|
|
var dateMenu = {
|
|
|
|
|
name: "date",
|
|
|
|
|
img: null,
|
|
|
|
|
items: [
|
|
|
|
|
{ name: "time",
|
|
|
|
|
get: () => ({ text: getTime(), img: null}),
|
|
|
|
|
show: function() { dateMenu.items[0].emit("redraw"); },
|
|
|
|
|
hide: function () {}
|
|
|
|
|
},
|
|
|
|
|
{ name: "date",
|
|
|
|
|
get: () => ({ text: getDate(), img: null}),
|
|
|
|
|
show: function() { dateMenu.items[1].emit("redraw"); },
|
|
|
|
|
hide: function () {}
|
|
|
|
|
},
|
|
|
|
|
{ name: "steps",
|
|
|
|
|
get: () => ({ text: Bangle.getHealthStatus("day").steps, img: null}),
|
|
|
|
|
show: function() { dateMenu.items[2].emit("redraw"); },
|
|
|
|
|
hide: function () {}
|
|
|
|
|
},
|
|
|
|
|
{ name: "battery",
|
|
|
|
|
get: () => ({ text: E.getBattery() + (Bangle.isCharging() ? "%++" : "%"), img: null}),
|
|
|
|
|
show: function() { dateMenu.items[3].emit("redraw"); },
|
|
|
|
|
hide: function () {}
|
|
|
|
|
},
|
|
|
|
|
]
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var menu = clock_info.load();
|
|
|
|
|
menu = menu.concat(dateMenu);
|
|
|
|
|
|
|
|
|
|
// Set draw functions for each item
|
|
|
|
|
var line = 0;
|
|
|
|
|
menu.forEach((menuItm, x) => {
|
|
|
|
|
menuItm.items.forEach((item, y) => {
|
|
|
|
|
function drawItem() {
|
|
|
|
|
item.hide();
|
|
|
|
|
|
|
|
|
|
var info = item.get();
|
|
|
|
|
drawText(item.name, info.text, (y%4)+1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
item.on('redraw', drawItem);
|
|
|
|
|
})
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Ensure that our settings are still in range (e.g. app uninstall). Otherwise reset the position it.
|
|
|
|
|
if(settings.menuPosX >= menu.length || settings.menuPosY > menu[settings.menuPosX].items.length ){
|
|
|
|
|
settings.menuPosX = 0;
|
|
|
|
|
settings.menuPosY = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function canRunMenuItem(){
|
|
|
|
|
var menuEntry = menu[settings.menuPosX];
|
|
|
|
|
var item = menuEntry.items[settings.menuPosY];
|
|
|
|
|
return item.run !== undefined;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function runMenuItem(){
|
|
|
|
|
var menuEntry = menu[settings.menuPosX];
|
|
|
|
|
var item = menuEntry.items[settings.menuPosY];
|
|
|
|
|
try{
|
|
|
|
|
var ret = item.run();
|
|
|
|
|
if(ret){
|
|
|
|
|
Bangle.buzz(300, 0.6);
|
|
|
|
|
}
|
|
|
|
|
} catch (ex) {
|
|
|
|
|
// Simply ignore it...
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
/************************************************
|
|
|
|
|
* Helper
|
|
|
|
|
*/
|
|
|
|
|
function getTime(){
|
|
|
|
|
var date = new Date();
|
|
|
|
|
return twoD(date.getHours())+ ":" + twoD(date.getMinutes());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getDate(){
|
|
|
|
|
var date = new Date();
|
|
|
|
|
return twoD(date.getDate()) + "." + twoD(date.getMonth());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
* Draw
|
|
|
|
|
*/
|
|
|
|
|
function draw() {
|
|
|
|
|
function draw() {
|
|
|
|
|
queueDraw();
|
|
|
|
|
|
|
|
|
|
g.clear();
|
|
|
|
|
Bangle.drawWidgets();
|
|
|
|
|
|
|
|
|
|
g.setFontUbuntuMono();
|
|
|
|
|
drawMainScreen();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function drawText(str, line, timeout){
|
|
|
|
|
timeout = timeout ? timeout : 0;
|
|
|
|
|
|
|
|
|
|
setTimeout(()=>{
|
|
|
|
|
g.setFontUbuntuMono();
|
|
|
|
|
var x = 10;
|
|
|
|
|
var y = line * 27 + 28;
|
|
|
|
|
g.setColor(g.theme.fg);
|
|
|
|
|
g.drawString(str, x, y);
|
|
|
|
|
}, timeout);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function drawLetters(str, line, offset){
|
|
|
|
|
offset= offset ? offset : 0;
|
|
|
|
|
var x = 10 + offset;
|
|
|
|
|
var y = line * 27 + 28;
|
|
|
|
|
|
|
|
|
|
var pos = 0;
|
|
|
|
|
for(var i=0; i < str.length; i++){
|
|
|
|
|
var c = str.charAt(i);
|
|
|
|
|
pos += g.stringWidth(c);
|
|
|
|
|
setTimeout((c, pos)=>{
|
|
|
|
|
g.setFontUbuntuMono();
|
|
|
|
|
g.setColor(g.theme.fg);
|
|
|
|
|
g.drawString(c, x+pos, y);
|
|
|
|
|
}, ts*(1+i), c, pos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function drawMainScreen(){
|
|
|
|
|
g.setFontUbuntuMono();
|
|
|
|
|
g.setFontAlign(-1, -1);
|
|
|
|
|
|
|
|
|
|
// Get menu item based on x
|
|
|
|
|
var menuItem = menu[settings.menuPosX];
|
|
|
|
|
var cmd = menuItem.name.slice(0,5).toLowerCase();
|
|
|
|
|
drawCmd(cmd);
|
|
|
|
|
|
|
|
|
|
// Draw menu items depending on our y value
|
|
|
|
|
drawMenuItems(menuItem);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function drawMenuItems(menuItem) {
|
|
|
|
|
var start = parseInt(settings.menuPosY / 4) * 4;
|
|
|
|
|
for (var i = start; i < start + 4; i++) {
|
|
|
|
|
if (i >= menuItem.items.length) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
lock_input++;
|
|
|
|
|
menuItem.items[i].show();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(!Bangle.isLocked()){
|
|
|
|
|
g.drawString(">", -2, ((settings.menuPosY % 4) + 1) * 27 + 28);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function drawText(key, value, line){
|
|
|
|
|
g.setFontUbuntuMono();
|
|
|
|
|
var x = 15;
|
|
|
|
|
var y = line * 27 + 28;
|
|
|
|
|
g.setColor(g.theme.fg);
|
|
|
|
|
|
|
|
|
|
function drawCmd(){
|
|
|
|
|
if(key){
|
|
|
|
|
key = (key.toLowerCase() + " ").slice(0, 4) + "|";
|
|
|
|
|
} else {
|
|
|
|
|
key = ""
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
value = String(value).replace("\n", " ");
|
|
|
|
|
g.drawString(key + value, x, y);
|
|
|
|
|
|
|
|
|
|
lock_input -= 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function drawCmd(cmd){
|
|
|
|
|
var c = 0;
|
|
|
|
|
var x = 10;
|
|
|
|
|
var y = 28;
|
|
|
|
@ -85,87 +208,140 @@ function drawCmd(){
|
|
|
|
|
|
|
|
|
|
g.setColor(g.theme.fg);
|
|
|
|
|
g.drawString("$", x+c, y);
|
|
|
|
|
c += g.stringWidth("$~");
|
|
|
|
|
}
|
|
|
|
|
c += g.stringWidth("$ ");
|
|
|
|
|
|
|
|
|
|
g.drawString(cmd, x+c, y);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function twoD(str){
|
|
|
|
|
return ("0" + str).slice(-2)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function drawMainScreen(){
|
|
|
|
|
var date = new Date();
|
|
|
|
|
// Draw time
|
|
|
|
|
|
|
|
|
|
g.setFontAlign(-1, -1);
|
|
|
|
|
var timeStr = ("0" + date.getHours()).slice(-2) + ":" + ("0" + date.getMinutes()).slice(-2);
|
|
|
|
|
var dateStr = locale.month(date, 1) + ("0" + date.getDate()).slice(-2);
|
|
|
|
|
drawCmd();
|
|
|
|
|
|
|
|
|
|
var cmd = "time";
|
|
|
|
|
var cmdT = ts * cmd.length + 50;
|
|
|
|
|
drawLetters("time", 0, 80);
|
|
|
|
|
drawText(" [t] " + timeStr, 1, cmdT);
|
|
|
|
|
drawText(" [d] " + dateStr, 2, cmdT);
|
|
|
|
|
drawText(" [" + (Bangle.isCharging() ? "c" : "b") + "] " + E.getBattery() + "%", 3, cmdT);
|
|
|
|
|
drawText(" [s] " + Bangle.getHealthStatus("day").steps, 4, cmdT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
/************************************************
|
|
|
|
|
* Listener
|
|
|
|
|
*/
|
|
|
|
|
// timeout used to update every minute
|
|
|
|
|
var drawTimeout;
|
|
|
|
|
// timeout used to update every minute
|
|
|
|
|
var drawTimeout;
|
|
|
|
|
|
|
|
|
|
// schedule a draw for the next minute
|
|
|
|
|
function queueDraw() {
|
|
|
|
|
// schedule a draw for the next minute
|
|
|
|
|
function queueDraw() {
|
|
|
|
|
if (drawTimeout) clearTimeout(drawTimeout);
|
|
|
|
|
drawTimeout = setTimeout(function() {
|
|
|
|
|
drawTimeout = undefined;
|
|
|
|
|
draw();
|
|
|
|
|
}, 60000 - (Date.now() % 60000));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Stop updates when LCD is off, restart when on
|
|
|
|
|
Bangle.on('lcdPower',on=>{
|
|
|
|
|
// Stop updates when LCD is off, restart when on
|
|
|
|
|
Bangle.on('lcdPower',on=>{
|
|
|
|
|
if (on) {
|
|
|
|
|
draw(); // draw immediately, queue redraw
|
|
|
|
|
} else { // stop draw timer
|
|
|
|
|
if (drawTimeout) clearTimeout(drawTimeout);
|
|
|
|
|
drawTimeout = undefined;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Bangle.on('lock', function(isLocked) {
|
|
|
|
|
if(isLocked){
|
|
|
|
|
Bangle.on('lock', function(isLocked) {
|
|
|
|
|
if (drawTimeout) clearTimeout(drawTimeout);
|
|
|
|
|
drawTimeout = undefined;
|
|
|
|
|
|
|
|
|
|
draw();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Bangle.on('charging',function(charging) {
|
|
|
|
|
if (drawTimeout) clearTimeout(drawTimeout);
|
|
|
|
|
drawTimeout = undefined;
|
|
|
|
|
|
|
|
|
|
draw();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
var lock_input = 0;
|
|
|
|
|
|
|
|
|
|
Bangle.on('touch', function(btn, e){
|
|
|
|
|
if(lock_input > 0){
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
lock_input = 0;
|
|
|
|
|
|
|
|
|
|
var left = parseInt(g.getWidth() * 0.22);
|
|
|
|
|
var right = g.getWidth() - left;
|
|
|
|
|
var upper = parseInt(g.getHeight() * 0.22) + 20;
|
|
|
|
|
var lower = g.getHeight() - upper;
|
|
|
|
|
|
|
|
|
|
var is_upper = e.y < upper;
|
|
|
|
|
var is_lower = e.y > lower;
|
|
|
|
|
var is_left = e.x < left && !is_upper && !is_lower;
|
|
|
|
|
var is_right = e.x > right && !is_upper && !is_lower;
|
|
|
|
|
var is_center = !is_upper && !is_lower && !is_left && !is_right;
|
|
|
|
|
|
|
|
|
|
if(is_lower){
|
|
|
|
|
if(settings.menuPosY >= menu[settings.menuPosX].items.length-1){
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (drawTimeout) clearTimeout(drawTimeout);
|
|
|
|
|
drawTimeout = undefined;
|
|
|
|
|
Bangle.buzz(40, 0.6);
|
|
|
|
|
settings.menuPosY++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(is_upper){
|
|
|
|
|
if(e.y < 20){ // Reserved for widget clicks
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(settings.menuPosY <= 0){
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
Bangle.buzz(40, 0.6);
|
|
|
|
|
settings.menuPosY--;
|
|
|
|
|
settings.menuPosY = settings.menuPosY < 0 ? 0 : settings.menuPosY;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(is_right){
|
|
|
|
|
Bangle.buzz(40, 0.6);
|
|
|
|
|
settings.menuPosX = (settings.menuPosX+1) % menu.length;
|
|
|
|
|
settings.menuPosY = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(is_left){
|
|
|
|
|
Bangle.buzz(40, 0.6);
|
|
|
|
|
settings.menuPosY = 0;
|
|
|
|
|
settings.menuPosX = settings.menuPosX-1;
|
|
|
|
|
settings.menuPosX = settings.menuPosX < 0 ? menu.length-1 : settings.menuPosX;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(is_center){
|
|
|
|
|
if(!canRunMenuItem()){
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
runMenuItem();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
draw();
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
E.on("kill", function(){
|
|
|
|
|
try{
|
|
|
|
|
storage.write(SETTINGS_FILE, settings);
|
|
|
|
|
} catch(ex){
|
|
|
|
|
// If this fails, we still kill the app...
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Bangle.on('charging',function(charging) {
|
|
|
|
|
if (drawTimeout) clearTimeout(drawTimeout);
|
|
|
|
|
drawTimeout = undefined;
|
|
|
|
|
|
|
|
|
|
draw();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************
|
|
|
|
|
/************************************************
|
|
|
|
|
* Startup Clock
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
// Show launcher when middle button pressed
|
|
|
|
|
Bangle.setUI("clock");
|
|
|
|
|
// Show launcher when middle button pressed
|
|
|
|
|
Bangle.setUI("clock");
|
|
|
|
|
|
|
|
|
|
// Load widgets and draw clock the first time
|
|
|
|
|
Bangle.loadWidgets();
|
|
|
|
|
// Load and draw widgets
|
|
|
|
|
Bangle.loadWidgets();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Draw first time
|
|
|
|
|
draw();
|
|
|
|
|
// Draw first time
|
|
|
|
|
draw();
|
|
|
|
|