You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
462 lines
12 KiB
462 lines
12 KiB
if (typeof Overlay === "undefined") {
|
|
var Overlay = class {
|
|
workout = null;
|
|
workoutState = null;
|
|
workoutStatus = null;
|
|
display = false;
|
|
milestones = [];
|
|
program = null;
|
|
currentCpm = null;
|
|
bikes = [];
|
|
programs = [];
|
|
workouts = [];
|
|
bike = null;
|
|
setup = null;
|
|
|
|
colors = {
|
|
"-2": "#F44",
|
|
"-1": "#FAA",
|
|
"0": "#FFF",
|
|
"1": "#AFA",
|
|
"2": "#4F4",
|
|
};
|
|
|
|
/**
|
|
* @param {HTMLElement} mBody
|
|
* @param {HTMLElement} mTopLeft
|
|
* @param {HTMLElement} mCenter
|
|
* @param {HTMLElement} mBottomRight
|
|
*/
|
|
constructor(mBody, mTopLeft, mCenter, mBottomRight) {
|
|
this.mBody = mBody;
|
|
this.mTopLeft = mTopLeft;
|
|
this.mCenter = mCenter;
|
|
this.mBottomRight = mBottomRight;
|
|
|
|
this.initStyle();
|
|
this.hide();
|
|
}
|
|
|
|
initStyle() {
|
|
this.applyDefaultStyle(this.mTopLeft);
|
|
this.applyDefaultStyle(this.mCenter);
|
|
this.applyDefaultStyle(this.mBottomRight);
|
|
|
|
this.mTopLeft.id = "overlay-mTopLeft";
|
|
this.mTopLeft.style.left = 0;
|
|
this.mTopLeft.style.top = "25%";
|
|
|
|
this.mCenter.id = "overlay-mCenter";
|
|
this.mCenter.style.left = "50%";
|
|
this.mCenter.style.top = "50%";
|
|
this.mCenter.style.transform = "translate(-50%, -50%)";
|
|
this.mCenter.style.backgroundColor = "rgba(0, 0, 0, 0.75)";
|
|
this.mCenter.style.fontSize = "4vmax";
|
|
|
|
this.mBottomRight.id = "overlay-mBottomRight";
|
|
this.mBottomRight.style.right = 0;
|
|
this.mBottomRight.style.bottom = 0;
|
|
this.mBottomRight.style.fontSize = "2vmax";
|
|
}
|
|
|
|
applyDefaultStyle(element) {
|
|
this.mBody.append(element);
|
|
|
|
element.style.position = "fixed";
|
|
element.style.zIndex = 99999999;
|
|
element.style.color = "#FFF";
|
|
element.style.backgroundColor = "rgba(0, 0, 0, 0.5)";
|
|
element.style.padding = "0.25em 0.5ch";
|
|
element.style.fontSize = "3vmax";
|
|
element.style.lineHeight = "1.33em";
|
|
}
|
|
|
|
hide() {
|
|
this.mTopLeft.style.display = "none";
|
|
this.mCenter.style.display = "none";
|
|
this.mBottomRight.style.display = "none";
|
|
|
|
this.display = false;
|
|
}
|
|
|
|
show() {
|
|
this.mTopLeft.style.display = "block";
|
|
this.mCenter.style.display = "block";
|
|
this.mBottomRight.style.display = "block";
|
|
|
|
this.display = true;
|
|
}
|
|
|
|
async setUp() {
|
|
this.writeCenter("Loading...");
|
|
|
|
this.bikes = await get("/bike");
|
|
this.programs = await get("/program");
|
|
this.workouts = await get("/workout?active=true");
|
|
|
|
this.writeCenter(null);
|
|
this.updateCenterState();
|
|
}
|
|
|
|
async updateCenterState() {
|
|
const state = this.getState();
|
|
|
|
switch (state) {
|
|
case "down":
|
|
this.writeCenter(`
|
|
(1) Ny økt<br/>
|
|
(2) Fortsett forrige økt
|
|
`);
|
|
break;
|
|
case "bike":
|
|
// TODO: Add this??
|
|
if (this.bikes.length === 1) {
|
|
this.setup = {state: "program", bike: this.bikes[0]};
|
|
this.updateCenterState();
|
|
} else {
|
|
this.writeCenter("Sykkelvalg ikke støttet");
|
|
}
|
|
break;
|
|
case "program":
|
|
const options = this.programs.map((p, i) => `(${i + 1}) ${p.name}`)
|
|
|
|
this.writeCenter("Velg programm:<br/>" + options.join("<br/>"));
|
|
break;
|
|
case "disconnected":
|
|
this.writeCenter("Kobler til...");
|
|
break;
|
|
case "connected":
|
|
this.writeCenter(`
|
|
(Enter) Start/Pause<br/>
|
|
(Esc) Stopp
|
|
`);
|
|
break;
|
|
case "started":
|
|
const calorieDiff = this.calorieDiff();
|
|
if (calorieDiff < 0) {
|
|
this.writeCenter(`
|
|
<big style="color: ${this.colors[-1]};"><big><b>${calorieDiff}</b></big> kcal!</big>
|
|
`);
|
|
} else {
|
|
this.writeCenter(null);
|
|
}
|
|
break;
|
|
default:
|
|
this.writeCenter(`Ugyldig tilstand: ${state}`);
|
|
}
|
|
|
|
this.updateKeyBindings();
|
|
}
|
|
|
|
updateKeyBindings() {
|
|
const dis = this;
|
|
|
|
this.mBody.onkeyup = function (event) {
|
|
const state = dis.getState();
|
|
|
|
if (state === "program") {
|
|
const i = parseInt(event.key, 10);
|
|
if (!isNaN(i) && typeof dis.programs[i - 1] !== "undefined") {
|
|
dis.bike = {...dis.setup.bike};
|
|
dis.startWorkout(dis.bike, dis.programs[i - 1]);
|
|
}
|
|
}
|
|
|
|
let handled = false;
|
|
switch (event.key) {
|
|
case "1":
|
|
if (state === "down") {
|
|
dis.setup = {state: "bike"};
|
|
handled = true;
|
|
}
|
|
break;
|
|
case "2":
|
|
if (state === "down") {
|
|
const last = dis.workouts[dis.workouts.length - 1];
|
|
console.log(last);
|
|
if (last !== undefined) {
|
|
dis.resumeWorkout(last);
|
|
}
|
|
|
|
handled = true;
|
|
}
|
|
break;
|
|
case "Enter":
|
|
if (state === "connected") {
|
|
dis.start();
|
|
handled = true;
|
|
} else if (state === "started") {
|
|
dis.pause();
|
|
handled = true;
|
|
}
|
|
break;
|
|
case "Escape":
|
|
if (state === "connected") {
|
|
dis.stop();
|
|
handled = true;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (handled) {
|
|
dis.updateCenterState();
|
|
event.preventDefault();
|
|
}
|
|
}
|
|
}
|
|
|
|
async startWorkout(bike, program) {
|
|
this.bike = bike;
|
|
this.program = program;
|
|
this.milestones = [];
|
|
|
|
const bikeId = bike.id;
|
|
const programId = program.id;
|
|
|
|
this.workout = await post("/workout", {bikeId, programId});
|
|
this.updateCenterState();
|
|
|
|
this.workout = await post(`/workout/${this.workout.id}/connect`);
|
|
this.updateCenterState();
|
|
|
|
this.setup = null;
|
|
this.webSocket();
|
|
}
|
|
|
|
async resumeWorkout(workout) {
|
|
this.bike = workout.bike;
|
|
this.program = workout.program;
|
|
this.workout = workout;
|
|
|
|
this.setup = null;
|
|
this.webSocket();
|
|
}
|
|
|
|
async start() {
|
|
this.workout = await post(`/workout/${this.workout.id}/start`);
|
|
this.updateCenterState();
|
|
}
|
|
|
|
async pause() {
|
|
this.workout = await post(`/workout/${this.workout.id}/pause`);
|
|
this.updateCenterState();
|
|
}
|
|
|
|
async stop() {
|
|
await post(`/workout/${this.workout.id}/stop`);
|
|
|
|
this.workout = null;
|
|
this.workouts = await post(`/workout?active=true`);
|
|
this.updateCenterState();
|
|
}
|
|
|
|
async webSocket() {
|
|
let dis = this;
|
|
|
|
this.socket = new WebSocket(url(`/workout/${this.workout.id}/subscribe`, "ws"));
|
|
|
|
this.socket.onmessage = function ({data, timeStamp}) {
|
|
let body = JSON.parse(data);
|
|
|
|
if (typeof body.workout !== "undefined") {
|
|
dis.workout = {...body.workout, ...dis.workout}
|
|
}
|
|
|
|
if (typeof body.workoutStatusBackfill !== "undefined") {
|
|
body.workoutStatusBackfill.forEach(wsbf => dis.updateWorkoutStatus(wsbf));
|
|
}
|
|
|
|
if (typeof body.workoutStatus !== "undefined") {
|
|
dis.updateWorkoutStatus(body.workoutStatus)
|
|
}
|
|
}
|
|
}
|
|
|
|
updateWorkoutStatus(newState = null) {
|
|
if (newState !== null) {
|
|
this.workoutStatus = newState;
|
|
}
|
|
|
|
if (this.workoutStatus === null) {
|
|
this.writeTopLeft("");
|
|
return;
|
|
}
|
|
|
|
const {minutes, seconds, calories, distance, rpm} = this.workoutStatus;
|
|
|
|
const cpmState = this.checkCpm();
|
|
const color = this.colors[cpmState];
|
|
|
|
this.writeTopLeft(`
|
|
<b><big>${pad(minutes)}</big></b> : <b><big>${pad(seconds)}</big></b><br/>
|
|
<b style="color: ${color}">${calories}</b> <small>kcal</small><br/>
|
|
<b>${distance.toFixed(1)}</b> <small>km</small><br/>
|
|
<b>${rpm}</b> <small>rpm</small><br/>
|
|
KPM: <b>${this.currentCpm.toFixed(1)}</b><br/>
|
|
`);
|
|
|
|
if (seconds === 0 && minutes > 0) {
|
|
this.updateMilestones(minutes, calories);
|
|
}
|
|
|
|
this.updateCenterState();
|
|
}
|
|
|
|
checkCpm() {
|
|
const {calories, minutes, seconds} = this.workoutStatus;
|
|
|
|
this.currentCpm = calories * 60 / ((minutes * 60) + seconds);
|
|
if (this.currentCpm === Infinity) {
|
|
this.currentCpm = 0;
|
|
}
|
|
|
|
const program = this.program;
|
|
if (program === null || this.currentCpm === 0) {
|
|
return 0;
|
|
}
|
|
|
|
if (this.calorieDiff() < 0) {
|
|
return -1;
|
|
} else {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
updateMilestones(minutes, calories) {
|
|
const expected = this.expectedCalories(minutes, 0);
|
|
const diff = calories - expected;
|
|
|
|
this.milestones.push({minutes, calories, diff});
|
|
|
|
if (minutes % 5 === 0) {
|
|
this.milestones = this.milestones.filter(m => m.minutes % 5 === 0)
|
|
}
|
|
|
|
let lines = "";
|
|
let lastDiff = 0;
|
|
this.milestones.sort((i, j) => j.minutes - i.minutes).forEach(({minutes, calories, diff}) => {
|
|
let color = this.colors["0"];
|
|
if (diff < 0) {
|
|
color = diff < lastDiff ? this.colors["-2"] : this.colors["-1"];
|
|
} else {
|
|
color = diff > lastDiff ? this.colors["2"] : this.colors["1"];
|
|
}
|
|
|
|
lines += `<tr style="text-align: right; color: ${color}"><td>${minutes} </td><td> ${calories}</td></tr>`;
|
|
});
|
|
|
|
if (lines.length > 0) {
|
|
this.writeBottomRight(`
|
|
<table>
|
|
${lines}
|
|
</table>
|
|
`);
|
|
} else {
|
|
this.writeBottomRight(null);
|
|
}
|
|
}
|
|
|
|
writeTopLeft(message) {
|
|
this.mTopLeft.innerHTML = message;
|
|
|
|
if (message === null || !this.display) {
|
|
this.mTopLeft.style.display = "none";
|
|
} else {
|
|
this.mTopLeft.style.display = "block";
|
|
}
|
|
}
|
|
|
|
writeCenter(message) {
|
|
this.mCenter.innerHTML = message;
|
|
|
|
if (message === null || !this.display) {
|
|
this.mCenter.style.display = "none";
|
|
} else {
|
|
this.mCenter.style.display = "block";
|
|
}
|
|
}
|
|
|
|
writeBottomRight(message) {
|
|
this.mBottomRight.innerHTML = message;
|
|
|
|
if (message === null || !this.display) {
|
|
this.mBottomRight.style.display = "none";
|
|
} else {
|
|
this.mBottomRight.style.display = "block";
|
|
}
|
|
}
|
|
|
|
calorieDiff() {
|
|
if (this.workoutStatus == null) {
|
|
return 0;
|
|
}
|
|
|
|
const {minutes, seconds, calories} = this.workoutStatus;
|
|
|
|
const expected = this.expectedCalories(minutes, seconds);
|
|
return calories - expected;
|
|
}
|
|
|
|
expectedCalories(minutes, seconds) {
|
|
const {warmupMin, warmupCpm, cpm} = this.program;
|
|
|
|
let preWarmup = 0;
|
|
if (warmupMin > 0) {
|
|
// Pre-warmup
|
|
const warmedUpMinutes = Math.min(minutes, warmupMin);
|
|
const warmedUpSeconds = minutes >= warmupMin ? 0 : seconds;
|
|
preWarmup = Math.round((warmupCpm * (warmedUpMinutes + (warmedUpSeconds / 60))));
|
|
}
|
|
|
|
// Post-warmup
|
|
const trainedMinutes = Math.max(0, minutes - warmupMin);
|
|
const trainedSeconds = minutes >= warmupMin ? seconds : 0;
|
|
const postWarmup = Math.round((cpm * (trainedMinutes + (trainedSeconds / 60))));
|
|
|
|
// Sum
|
|
return Math.round(preWarmup + postWarmup);
|
|
}
|
|
|
|
getState() {
|
|
return (this.workout || this.setup || {state: "down"}).state;
|
|
}
|
|
};
|
|
}
|
|
|
|
function pad(number) {
|
|
return number >= 10 ? `${number}` : `0${number}`;
|
|
}
|
|
|
|
function url(path, prefix = "http") {
|
|
return `${prefix}://127.0.0.1:9999/api${path}`;
|
|
}
|
|
|
|
function get(path) {
|
|
return fetch(url(path), {
|
|
method: "GET",
|
|
}).then(r => r.json());
|
|
}
|
|
|
|
function post(path, data = {}) {
|
|
return fetch(url(path), {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
},
|
|
body: JSON.stringify(data)
|
|
}).then(r => r.json());
|
|
}
|
|
|
|
|
|
const overlay =
|
|
new Overlay(
|
|
document.body,
|
|
document.createElement("div"),
|
|
document.createElement("div"),
|
|
document.createElement("div"));
|
|
|
|
overlay.show();
|
|
overlay.setUp();
|
|
|
|
chrome.storage.sync.get('color', ({color}) => {
|
|
});
|