garmin-repafield/source/RepaFieldView.mc

719 lines
24 KiB
MonkeyC

import Toybox.Activity;
import Toybox.FitContributor;
import Toybox.Graphics;
import Toybox.Lang;
import Toybox.UserProfile;
import Toybox.WatchUi;
import Toybox.System;
import Toybox.Application;
const HR_TYPE_PERCENT = 1;
const HR_TYPE_ZONE = 2;
const TLF_CADENCE = 0;
const TLF_GRADE = 1;
const TLF_GAP = 2;
const TLF_VSPEED = 3;
const FIT_GRADE_ID = 0;
const FIT_GRADE_SUM_MIN_ID = 5;
const FIT_GRADE_SUM_MAX_ID = 6;
const FIT_GRADE_SUM_AVG_ID = 7;
const FIT_GRADE_LAP_AVG_ID = 8;
const FIT_GAP_ID = 1;
const FIT_GAP_SUM_AVG_ID = 3;
const FIT_GAP_LAP_AVG_ID = 4;
const FIT_VSPEED_ID = 2;
const FIT_VSPEED_SUM_MIN_ID = 9;
const FIT_VSPEED_SUM_MAX_ID = 10;
const FIT_VSPEED_SUM_AVG_ID = 11;
const FIT_VSPEED_LAP_AVG_ID = 12;
function displayHr(hr as Number, type as Number, zones as Array<Number>) as String {
if (hr == 0) {
return "-";
} else if (type == 1) {
var maxHr = zones[zones.size() - 1];
var percentage = (hr.toFloat() / maxHr) * 100;
return percentage.format("%.0f") + "%";
} else if (type == 2) {
var hrzsize = zones.size();
for (var i = 0; i < hrzsize; i++) {
if (i == 0 && hr < zones[i]) {
return "-";
} else if (hr < zones[i] || i == hrzsize - 1) {
return ((hr - zones[i - 1]) / (zones[i] - zones[i - 1]).toFloat() + i).format("%.1f");
}
}
} else { // type == 0 or anything else
return hr.format("%d");
}
return "?";
}
class RepaFieldView extends WatchUi.DataField {
// settings
hidden var themeColor as Number;
hidden var themeColor2 as Number;
hidden var themeColor3 as Number;
hidden var hrDisplayType as Number;
hidden var speedNotPace as Boolean;
hidden var showNextPoint as Boolean;
hidden var tlFieldData as Number;
hidden var hrZones as Array<Number>;
hidden var hrHist as Array<Number>;
hidden var hrZoneColors as Array<Number>;
hidden var cadenceZones as Array<Number>;
hidden var cadenceZoneColors as Array<Number>;
hidden var gradeZones as Array<Number>;
hidden var gradeZoneColors as Array<Number>;
hidden var vsZones as Array<Number>;
hidden var vsZoneColors as Array<Number>;
hidden var isDistanceMetric as Boolean;
hidden var isElevationMetric as Boolean;
hidden var isPaceMetric as Boolean;
hidden var mileToKm as Float = 1.609344f;
hidden var meterToFeet as Float = 3.28084f;
// fields
hidden var fBgOverlay;
hidden var fTrack;
hidden var fPace;
hidden var fAPace;
hidden var fElevation;
hidden var fElevationGain;
hidden var fElevationLoss;
hidden var fCadence;
hidden var fDistance;
hidden var fTime;
hidden var fTimer;
hidden var fTimerSec;
hidden var fHr;
hidden var fAHr;
hidden var fMHr;
hidden var fHrGraph;
// fit
hidden var fitGrade;
hidden var fitGradeSumAvg;
hidden var fitGradeLapAvg;
hidden var fitGAP;
hidden var fitVSpeed;
hidden var fitVSpeedSumAvg;
hidden var fitVSpeedLapAvg;
// values
hidden var timerRunning as Boolean = false;
hidden var hrTicks as Number;
hidden var hrValue as Numeric;
hidden var ahrValue as Numeric;
hidden var mhrValue as Numeric;
hidden var toDestination as Float;
hidden var toNextPoint as Float;
hidden var nextPointName as String;
hidden var distance as Float;
hidden var timer as Numeric;
hidden var timerState as Number;
hidden var offCourse as Float;
hidden var speed as Float;
hidden var aspeed as Float;
hidden var pace as Float;
hidden var apace as Float;
hidden var altitude as Float;
hidden var egain as Number;
hidden var edrop as Number;
hidden var cadence as Number;
hidden var grade as RollingAverage;
hidden var cgrade;
hidden var vspeed as RollingAverage;
hidden var cvspeed;
hidden var cgap;
function initialize() {
DataField.initialize();
themeColor = Application.Properties.getValue("themeColor").toNumberWithBase(16);
themeColor2 = Application.Properties.getValue("themeColor2").toNumberWithBase(16);
themeColor3 = Application.Properties.getValue("themeColor3").toNumberWithBase(16);
hrDisplayType = Application.Properties.getValue("hrDisplay").toNumber();
speedNotPace = Application.Properties.getValue("speedNotPace");
showNextPoint = Application.Properties.getValue("showNextPoint");
tlFieldData = Application.Properties.getValue("tlFieldData").toNumber();
hrValue = 0;
ahrValue = 0;
mhrValue = 0;
hrZones = UserProfile.getHeartRateZones(UserProfile.getCurrentSport());
hrHist = [0, 0, 0, 0, 0, 0, 0];
hrTicks = 0;
hrZoneColors = [Graphics.COLOR_DK_GRAY, Graphics.COLOR_LT_GRAY, Graphics.COLOR_BLUE, Graphics.COLOR_GREEN, Graphics.COLOR_YELLOW, Graphics.COLOR_RED, Graphics.COLOR_DK_RED];
cadenceZones = [153, 163, 173, 183];
cadenceZoneColors = [Graphics.COLOR_RED, Graphics.COLOR_YELLOW, Graphics.COLOR_GREEN, Graphics.COLOR_BLUE, Graphics.COLOR_PURPLE];
gradeZones = [-10, -1, 1, 3, 6, 10, 15];
gradeZoneColors = [Graphics.COLOR_PINK, Graphics.COLOR_PURPLE, Graphics.COLOR_LT_GRAY, Graphics.COLOR_BLUE, Graphics.COLOR_GREEN, Graphics.COLOR_YELLOW, Graphics.COLOR_RED, Graphics.COLOR_DK_RED];
vsZones = [-16.6, -1.6, 1.6, 5.0, 10.0, 16.6, 25.0];
vsZoneColors = gradeZoneColors;
toDestination = 0.0f;
toNextPoint = 0.0f;
nextPointName = "";
distance = 0.0f;
timer = 0;
timerState = Activity.TIMER_STATE_OFF;
offCourse = 0.0f;
speed = 0.0f;
aspeed = 0.0f;
pace = 0.0f;
apace = 0.0f;
altitude = 0.0f;
egain = 0;
edrop = 0;
cadence = 0;
grade = new RollingAverage(10);
cgrade = null;
vspeed = new RollingAverage(10);
cvspeed = null;
cgap = null;
var settings = System.getDeviceSettings();
isDistanceMetric = settings.distanceUnits == System.UNIT_METRIC;
isElevationMetric = settings.elevationUnits == System.UNIT_METRIC;
isPaceMetric = settings.paceUnits == System.UNIT_METRIC;
// fit fields
// TODO: refator into separate function/class
fitGrade = null;
fitGradeSumAvg = null;
fitGradeLapAvg = null;
fitGAP = null;
fitVSpeed = null;
fitVSpeedSumAvg = null;
fitVSpeedLapAvg = null;
if (Application.Properties.getValue("saveToFit")) {
fitGrade = DataField.createField(
"grade",
FIT_GRADE_ID,
FitContributor.DATA_TYPE_FLOAT,
{
:mesgType=>FitContributor.MESG_TYPE_RECORD,
:units=>"%",
}
);
fitGradeSumAvg = DataField.createField(
"avg_grade",
FIT_GRADE_SUM_AVG_ID,
FitContributor.DATA_TYPE_FLOAT,
{
:mesgType=>FitContributor.MESG_TYPE_RECORD,
:units=>"%",
}
);
fitGradeLapAvg = DataField.createField(
"avg_grade",
FIT_GRADE_LAP_AVG_ID,
FitContributor.DATA_TYPE_FLOAT,
{
:mesgType=>FitContributor.MESG_TYPE_RECORD,
:units=>"%",
}
);
fitGAP = DataField.createField(
"gap",
FIT_GAP_ID,
FitContributor.DATA_TYPE_FLOAT,
{
:mesgType=>FitContributor.MESG_TYPE_RECORD,
:units=>isPaceMetric ? "min/km" : "min/mi",
}
);
fitVSpeed = DataField.createField(
"vspeed",
FIT_VSPEED_ID,
FitContributor.DATA_TYPE_FLOAT,
{
:mesgType=>FitContributor.MESG_TYPE_RECORD,
:units=>isElevationMetric ? "m/min" : "ft/min",
}
);
fitVSpeedSumAvg = DataField.createField(
"avg_vspeed",
FIT_VSPEED_SUM_AVG_ID,
FitContributor.DATA_TYPE_FLOAT,
{
:mesgType=>FitContributor.MESG_TYPE_RECORD,
:units=>isElevationMetric ? "m/min" : "ft/min",
}
);
fitVSpeedLapAvg = DataField.createField(
"avg_vspeed",
FIT_VSPEED_LAP_AVG_ID,
FitContributor.DATA_TYPE_FLOAT,
{
:mesgType=>FitContributor.MESG_TYPE_RECORD,
:units=>isElevationMetric ? "m/min" : "ft/min",
}
);
}
}
public function onNextMultisportLeg() as Void {
grade.reset();
vspeed.reset();
}
public function onTimerLap() as Void {
grade.lapReset();
vspeed.lapReset();
}
public function onTimerReset() as Void {
grade.reset();
vspeed.reset();
}
public function onTimerStart() as Void {
timerRunning = true;
}
public function onTimerResume() as Void {
timerRunning = true;
}
public function onTimerPause() as Void {
timerRunning = false;
}
public function onTimerStop() as Void {
timerRunning = false;
}
function tickHr(v as Number) {
hrTicks++;
var hrzsize = hrZones.size();
for (var i = 0; i < hrzsize; i++) {
if (v < hrZones[i]) {
hrHist[i]++;
return;
}
}
// out of range
hrHist[hrzsize]++;
}
function calculateZoneColor(v as Numeric, zones as Array<Numeric>, zoneColors as Array<Numeric>) as Numeric {
for (var i = 0; i < zones.size(); i++) {
if (v < zones[i]) {
return zoneColors[i];
}
}
return zoneColors[zoneColors.size() - 1];
}
function darken(color as Numeric, factor as Numeric) as Numeric {
var r = (color >> 16) & 0xFF;
var g = (color >> 8) & 0xFF;
var b = color & 0xFF;
r = r / factor.toFloat();
g = g / factor.toFloat();
b = b / factor.toFloat();
return (r.toLong() << 16) | (g.toLong() << 8) | b.toLong();
}
// Set your layout here. Anytime the size of obscurity of
// the draw context is changed this will be called.
function onLayout(dc as Dc) as Void {
var obscurityFlags = DataField.getObscurityFlags();
// Top layout
if (obscurityFlags == (OBSCURE_TOP | OBSCURE_LEFT | OBSCURE_RIGHT)) {
View.setLayout(Rez.Layouts.TopLayout(dc));
return;
// Bottom layout
} else if (obscurityFlags == (OBSCURE_BOTTOM | OBSCURE_LEFT | OBSCURE_RIGHT)) {
View.setLayout(Rez.Layouts.BottomLayout(dc));
return;
}
// Use the generic, centered layout
View.setLayout(Rez.Layouts.MainLayout(dc));
// fields
fBgOverlay = View.findDrawableById("BgOverlay") as BgOverlay;
fTrack = View.findDrawableById("Track") as Track;
fPace = View.findDrawableById("pace") as Text;
fAPace = View.findDrawableById("apace") as Text;
fElevation = View.findDrawableById("elevation") as Text;
fElevationGain = View.findDrawableById("elevationGain") as Text;
fElevationLoss = View.findDrawableById("elevationLoss") as Text;
fTime = View.findDrawableById("time") as Text;
fTimer = View.findDrawableById("timerHM") as Text;
fTimerSec = View.findDrawableById("timerS") as Text;
fCadence = View.findDrawableById("cadence") as Text;
fDistance = View.findDrawableById("distance") as Text;
fHr = View.findDrawableById("hr") as Text;
fAHr = View.findDrawableById("ahr") as Text;
fMHr = View.findDrawableById("mhr") as Text;
fHrGraph = View.findDrawableById("HeartRate") as HeartRate;
// init fields
fHrGraph.setHRZoneColors(hrZoneColors);
// theme setup
if (themeColor != 0) {
fBgOverlay.setColor1(darken(themeColor, 4));
fBgOverlay.setColor2(darken(themeColor, 2));
fAPace.setColor(themeColor);
} else {
fAPace.setColor(Graphics.COLOR_WHITE);
}
fElevation.setColor(themeColor2);
fElevationGain.setColor(themeColor2);
fElevationLoss.setColor(themeColor2);
fTime.setColor(themeColor3);
// units
if (!isDistanceMetric) {
var fdl = View.findDrawableById("distanceLabel") as Text;
fdl.setText("mi");
}
}
function compute(info as Activity.Info) as Void {
// update rolling values before updating normal fields
// only calculate them when some time has passed
if (info.timerTime != null && info.timerTime > 0 && timerRunning) {
if (info.altitude != null) {
var altChange = info.altitude - altitude;
// grade
if (info.elapsedDistance != null) {
var distChange = info.elapsedDistance - distance;
if (distChange > 0) {
grade.insert(altChange / distChange);
}
var currentGrade = grade.getRolling();
if (currentGrade) {
cgrade = currentGrade * 100;
}
}
// vspeed - m/min or ft/min
var timerChange = info.timerTime - timer;
if (timerChange > 0) {
if (!isElevationMetric) {
vspeed.insert(meterToFeet * altChange / (timerChange / 60000.0));
} else {
vspeed.insert(altChange / (timerChange / 60000.0));
}
var currentVSpeed = vspeed.getRolling();
if (currentVSpeed != null ) {
cvspeed = currentVSpeed;
}
}
}
}
// update normal fields
if (info.currentHeartRate != null) {
hrValue = info.currentHeartRate as Number;
tickHr(hrValue);
} else {
hrValue = 0;
}
if (info.averageHeartRate != null) {
ahrValue = info.averageHeartRate as Number;
} else {
ahrValue = 0;
}
if (info.maxHeartRate != null) {
mhrValue = info.maxHeartRate as Number;
} else {
mhrValue = 0;
}
if (info.elapsedDistance != null) {
distance = info.elapsedDistance as Float;
} else {
distance = 0.0f;
}
if (info.timerTime != null) {
timer = info.timerTime;
} else {
timer = 0;
}
if (info.timerState != null) {
timerState = info.timerState;
} else {
timerState = Activity.TIMER_STATE_OFF;
}
if (info has :distanceToDestination && info.distanceToDestination != null) {
toDestination = info.distanceToDestination as Float;
} else {
toDestination = 0.0f;
}
if (info has :distanceToNextPoint && info.distanceToNextPoint != null) {
toNextPoint = info.distanceToNextPoint as Float;
} else {
toNextPoint = 0.0f;
}
if (info has :nameOfNextPoint && info.nameOfNextPoint != null) {
nextPointName = info.nameOfNextPoint as String;
} else {
nextPointName = "";
}
if (info has :offCourseDistance && info.offCourseDistance != null) {
offCourse = info.offCourseDistance as Float;
} else {
offCourse = 0.0f;
}
if (info.currentSpeed != null) {
speed = info.currentSpeed * 3.6;
if (speed == 0) {
pace = 0.0f;
} else {
pace = 60 / speed;
}
} else {
speed = 0.0f;
pace = 0.0f;
}
if (info.averageSpeed != null) {
aspeed = info.averageSpeed * 3.6;
if (aspeed == 0) {
apace = 0.0f;
} else {
apace = 60 / aspeed;
}
} else {
aspeed = 0.0f;
apace = 0.0f;
}
if (info.altitude != null) {
altitude = info.altitude as Float;
} else {
altitude = 0.0f;
}
if (info.totalAscent != null) {
egain = info.totalAscent as Number;
} else {
egain = 0;
}
if (info.totalDescent != null) {
edrop = info.totalDescent as Number;
} else {
edrop = 0;
}
if (info.currentCadence != null) {
cadence = info.currentCadence as Number;
} else {
cadence = 0;
}
if (cgrade != null) {
cgap = adjustPaceForGrade(pace, cgrade / 100);
}
// convert units to imperial if needed
if (!isDistanceMetric) {
distance = distance / mileToKm;
toDestination = toDestination / mileToKm;
}
if (!isElevationMetric) {
altitude = altitude * meterToFeet;
egain = (egain * meterToFeet).toNumber();
edrop = (edrop * meterToFeet).toNumber();
}
if (!isPaceMetric) {
pace = pace * mileToKm;
apace = apace * mileToKm;
speed = speed / mileToKm;
aspeed = aspeed / mileToKm;
}
// fit update
// TODO: refactor into separate function/class
if (timerRunning) {
if (fitGrade != null) {
fitGrade.setData(cgrade ? cgrade : 0);
var gradeSumAvg = grade.totalAvg();
fitGradeSumAvg.setData(gradeSumAvg ? gradeSumAvg : 0);
var gradeLapAvg = grade.lapAvg();
fitGradeLapAvg.setData(gradeLapAvg ? gradeLapAvg : 0);
}
if (fitGAP != null) {
fitGAP.setData(cgap ? cgap : 0);
}
if (fitVSpeed != null) {
fitVSpeed.setData(cvspeed ? cvspeed : 0);
var vsSumAvg = grade.totalAvg();
fitVSpeedSumAvg.setData(vsSumAvg ? vsSumAvg : 0);
var vsLapAvg = grade.lapAvg();
fitVSpeedLapAvg.setData(vsLapAvg ? vsLapAvg : 0);
}
}
}
// TODO: onlap - reset lap metrics
// Display the value you computed here. This will be called
// once a second when the data field is visible.
function onUpdate(dc as Dc) as Void {
// HR value
var hrColor = calculateZoneColor(hrValue, hrZones, hrZoneColors);
fHr.setColor(hrColor);
fAHr.setColor(darken(calculateZoneColor(ahrValue, hrZones, hrZoneColors), 2));
fMHr.setColor(darken(calculateZoneColor(mhrValue, hrZones, hrZoneColors), 2));
fHr.setText(displayHr(hrValue, hrDisplayType, hrZones));
fAHr.setText(displayHr(ahrValue, hrDisplayType, hrZones));
fMHr.setText(displayHr(mhrValue, hrDisplayType, hrZones));
if (fHrGraph != null) {
fHrGraph.setHRHist(hrHist);
fHrGraph.setHRTicks(hrTicks);
}
if (themeColor == 0) {
fBgOverlay.setColor1(darken(hrColor, 4));
fBgOverlay.setColor2(darken(hrColor, 2));
}
// track
if (fTrack != null) {
fTrack.setTrackData(toDestination, toNextPoint, distance, offCourse, nextPointName, showNextPoint);
}
// time
if (fTime != null) {
var time = System.getClockTime();
var tstr = time.hour.format("%02d") + ":" + time.min.format("%02d");
fTime.setText(tstr);
}
// timer
if (fTimer != null) {
var timersec = timer / 1000;
var trh = timersec / 3600;
var trm = (timersec % 3600) / 60;
var trs = timersec % 60;
var timerColor = Graphics.COLOR_RED;
if (timerState == Activity.TIMER_STATE_ON) {
timerColor = Graphics.COLOR_WHITE;
} else if (timerState == Activity.TIMER_STATE_PAUSED) {
timerColor = Graphics.COLOR_YELLOW;
}
fTimer.setColor(timerColor);
if (trh > 0) {
fTimer.setText(trh.format("%d") + ":" + trm.format("%02d"));
} else {
fTimer.setText(trm.format("%02d"));
}
if (fTimerSec != null) {
fTimerSec.setColor(darken(timerColor, 1.5));
fTimerSec.setText(":" + trs.format("%02d"));
}
}
// distance
if (fDistance != null) {
if (distance >= 100000) {
fDistance.setText((distance / 1000).format("%.0f"));
} else if (distance >= 10000) {
fDistance.setText((distance / 1000).format("%.1f"));
} else {
fDistance.setText((distance / 1000).format("%.2f"));
}
}
// pace
if (fPace != null) {
if (speedNotPace) {
fPace.setText(speed.format("%.1f"));
} else {
if (speed != 0) {
var pmin = pace.toNumber();
var psec = (pace - pmin) * 60;
fPace.setText(pmin.format("%2d") + ":" + psec.format("%02d"));
} else {
fPace.setText("--:--");
}
}
}
if (fAPace != null) {
if (speedNotPace) {
fAPace.setText(aspeed.format("%.1f"));
} else {
if (aspeed != 0) {
var apmin = apace.toNumber();
var apsec = (apace - apmin) * 60;
fAPace.setText(apmin.format("%2d") + ":" + apsec.format("%02d"));
} else {
fAPace.setText("--:--");
}
}
}
// alt/egain/edrop
if (fElevation != null) {
// draw icon
fElevation.setText(altitude.format("%.0f"));
}
if (fElevationGain != null) {
fElevationGain.setText(egain.format("%d"));
}
if (fElevationLoss != null) {
fElevationLoss.setText(edrop.format("%d"));
}
// TLF
if (fCadence != null) {
if (tlFieldData == TLF_GRADE) {
if (cgrade != null) {
var gradeColor = calculateZoneColor(cgrade, gradeZones, gradeZoneColors);
fCadence.setColor(gradeColor);
if (cgrade >= 10 || cgrade <= -10) {
fCadence.setText(cgrade.format("%.0f"));
} else {
fCadence.setText(cgrade.format("%.1f"));
}
} else {
fCadence.setText("-");
}
} else if (tlFieldData == TLF_GAP) {
fCadence.setColor(themeColor2);
if (pace != 0 && cgap != null) {
// TODO color
var gapmin = cgap.toNumber();
var gapsec = (cgap - gapmin) * 60;
fCadence.setText(gapmin.format("%d") + ":" + gapsec.format("%02d"));
} else {
fCadence.setText("-");
}
} else if (tlFieldData == TLF_VSPEED) {
if (cvspeed != null) {
var vsColor = calculateZoneColor(cvspeed, vsZones, vsZoneColors);
fCadence.setColor(vsColor);
if (cvspeed >= 10 || cvspeed <= -10) {
fCadence.setText(cvspeed.format("%.0f"));
} else {
fCadence.setText(cvspeed.format("%.1f"));
}
} else {
fCadence.setText("-");
}
} else {
var cadenceColor = calculateZoneColor(cadence, cadenceZones, cadenceZoneColors);
fCadence.setColor(cadenceColor);
if (cadence != 0) {
fCadence.setText(cadence.format("%d"));
} else {
fCadence.setText("-");
}
}
}
// Call parent's onUpdate(dc) to redraw the layout
View.onUpdate(dc);
}
}