1
0
mirror of https://codeberg.org/Freeyourgadget/Gadgetbridge synced 2025-01-24 16:47:32 +01:00

Bangle.js: activity tracks summary entries

This commit is contained in:
Ganblejs 2023-11-07 02:46:11 +01:00 committed by José Rebelo
parent eec3d2b89a
commit e10fae00cf

View File

@ -801,10 +801,11 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
// Calculate and store analytical data (distance, speed, cadence, etc.). // Calculate and store analytical data (distance, speed, cadence, etc.).
JSONObject analyticsObject = new JSONObject(); JSONObject analyticsObject = new JSONObject();
JSONArray calculationsArray = new JSONArray(); JSONArray calculationsArray = new JSONArray();
int logLength = storedLogObject.getJSONArray("Time").length();
// Add elapsed time since first reading (seconds). // Add elapsed time since first reading (seconds).
valueArray = storedLogObject.getJSONArray("Time"); valueArray = storedLogObject.getJSONArray("Time");
for (int i = 0; i < valueArray.length(); i++) { for (int i = 0; i < logLength; i++) {
calculationsArray.put(valueArray.getDouble(i)-valueArray.getDouble(0)); calculationsArray.put(valueArray.getDouble(i)-valueArray.getDouble(0));
} }
analyticsObject.put("Elapsed Time", calculationsArray); analyticsObject.put("Elapsed Time", calculationsArray);
@ -819,7 +820,7 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
// Add distance between last and current reading. // Add distance between last and current reading.
valueArray = storedLogObject.getJSONArray("Latitude"); valueArray = storedLogObject.getJSONArray("Latitude");
valueArray2 = storedLogObject.getJSONArray("Longitude"); valueArray2 = storedLogObject.getJSONArray("Longitude");
for (int i = 0; i < valueArray.length(); i++) { for (int i = 0; i < logLength; i++) {
if (i == 0) { if (i == 0) {
calculationsArray.put("0"); calculationsArray.put("0");
} else { } else {
@ -838,14 +839,31 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
valueArray2 = new JSONArray(); valueArray2 = new JSONArray();
calculationsArray = new JSONArray(); calculationsArray = new JSONArray();
// Add stride lengths between consecutive readings.
if (storedLogObject.has("Steps")) {
for (int i = 0; i < logLength; i++) {
if (i == 0) {
calculationsArray.put(null);
} else {
double calculation =
2 * analyticsObject.getJSONArray("Intermediate Distance").getDouble(i) /
storedLogObject.getJSONArray("Steps").getDouble(i);
calculationsArray.put(calculation);
}
}
analyticsObject.put("Stride", calculationsArray);
calculationsArray = new JSONArray();
}
} else if (storedLogObject.has("Steps")) { } else if (storedLogObject.has("Steps")) {
for (int i = 0; i < storedLogObject.getJSONArray("Steps").length(); i++) { for (int i = 0; i < logLength; i++) {
if (i==0) { if (i==0) {
calculationsArray.put(0); calculationsArray.put(0);
} else { } else {
double stride = 0.85; // FIXME: Depend on user defined stride length? double stride = 0.85; // TODO: Depend on user defined stride length?
double calculation = stride * (storedLogObject.getJSONArray("Steps").getDouble(i)); double calculation = stride * (storedLogObject.getJSONArray("Steps").getDouble(i));
if (calculation == 0) calculation = 0.001; // To avoid potential division by zero later on. //if (calculation == 0) calculation = 0.001; // To avoid potential division by zero later on.
calculationsArray.put(calculation); calculationsArray.put(calculation);
} }
} }
@ -857,7 +875,7 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
if (analyticsObject.has("Intermediate Distance")) { if (analyticsObject.has("Intermediate Distance")) {
// Add total distance from start of activity up to each reading. // Add total distance from start of activity up to each reading.
for (int i = 0; i < analyticsObject.getJSONArray("Intermediate Distance").length(); i++) { for (int i = 0; i < logLength; i++) {
if (i==0) { if (i==0) {
calculationsArray.put(0); calculationsArray.put(0);
} else { } else {
@ -870,7 +888,7 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
calculationsArray = new JSONArray(); calculationsArray = new JSONArray();
// Add average speed between last and current reading (m/s). // Add average speed between last and current reading (m/s).
for (int i = 0; i < analyticsObject.getJSONArray("Intermediate Distance").length(); i++) { for (int i = 0; i < logLength; i++) {
if (i==0) { if (i==0) {
calculationsArray.put(0); calculationsArray.put(0);
} else { } else {
@ -887,8 +905,8 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
calculationsArray = new JSONArray(); calculationsArray = new JSONArray();
// Add average pace between last and current reading (s/km). (Was gonna do this as min/km but summary seem to expect s/km). // Add average pace between last and current reading (s/km). (Was gonna do this as min/km but summary seems to expect s/km).
for (int i = 0; i < analyticsObject.getJSONArray("Speed").length(); i++) { for (int i = 0; i < logLength; i++) {
if (i==0) { if (i==0) {
calculationsArray.put(0); calculationsArray.put(0);
} else { } else {
@ -899,9 +917,28 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
analyticsObject.put("Pace", calculationsArray); analyticsObject.put("Pace", calculationsArray);
calculationsArray = new JSONArray(); calculationsArray = new JSONArray();
} }
if (storedLogObject.has("Steps")) {
for (int i = 0; i < logLength; i++) {
if (i==0) {
calculationsArray.put(0);
} else {
double calculation = 60 *
(storedLogObject.getJSONArray("Steps").getDouble(i) /
(storedLogObject.getJSONArray("Time").getDouble(i) -
storedLogObject.getJSONArray("Time").getDouble(i-1)
)
);// Should cadence be steps/min or half that? https://www.polar.com/blog/what-is-running-cadence/
calculationsArray.put(calculation);
}
}
analyticsObject.put("Cadence", calculationsArray);
calculationsArray = new JSONArray();
}
LOG.info("AnalyticsObject:\n" + analyticsObject);
BaseActivitySummary summary = null; BaseActivitySummary summary = null;
@ -932,61 +969,87 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
// "baseAltitude", "ascentSeconds", "descentSeconds", "flatSeconds", "ascentDistance", // "baseAltitude", "ascentSeconds", "descentSeconds", "flatSeconds", "ascentDistance",
// "descentDistance", "flatDistance", "elevationGain", "elevationLoss" // "descentDistance", "flatDistance", "elevationGain", "elevationLoss"
// )); // ));
summaryData = addSummaryData(summaryData,"ascentMeters",3,"mm"); if (storedLogObject.has("Altitude") || storedLogObject.has("Barometer Altitude")) {
summaryData = addSummaryData(summaryData,"descentMeters",3,"mm"); summaryData = addSummaryData(summaryData, "ascentMeters", 3, "mm");
summaryData = addSummaryData(summaryData,"maxAltitude",3,"mm"); summaryData = addSummaryData(summaryData, "descentMeters", 3, "mm");
summaryData = addSummaryData(summaryData,"minAltitude",3,"mm"); summaryData = addSummaryData(summaryData, "maxAltitude", 3, "mm");
summaryData = addSummaryData(summaryData,"averageAltitude",3,"mm"); summaryData = addSummaryData(summaryData, "minAltitude", 3, "mm");
summaryData = addSummaryData(summaryData,"baseAltitude",3,"mm"); summaryData = addSummaryData(summaryData, "averageAltitude", 3, "mm");
summaryData = addSummaryData(summaryData,"ascentSeconds",3,"mm"); summaryData = addSummaryData(summaryData, "baseAltitude", 3, "mm");
summaryData = addSummaryData(summaryData,"descentSeconds",3,"mm"); summaryData = addSummaryData(summaryData, "ascentSeconds", 3, "mm");
summaryData = addSummaryData(summaryData,"flatSeconds",3,"mm"); summaryData = addSummaryData(summaryData, "descentSeconds", 3, "mm");
summaryData = addSummaryData(summaryData,"ascentDistance",3,"mm"); summaryData = addSummaryData(summaryData, "flatSeconds", 3, "mm");
summaryData = addSummaryData(summaryData,"descentDistance",3,"mm"); if (analyticsObject.has("Intermittent Distance")) {
summaryData = addSummaryData(summaryData,"flatDistance",3,"mm"); summaryData = addSummaryData(summaryData, "ascentDistance", 3, "mm");
summaryData = addSummaryData(summaryData,"elevationGain",3,"mm"); summaryData = addSummaryData(summaryData, "descentDistance", 3, "mm");
summaryData = addSummaryData(summaryData,"elevationLoss",3,"mm"); summaryData = addSummaryData(summaryData, "flatDistance", 3, "mm");
}
summaryData = addSummaryData(summaryData, "elevationGain", 3, "mm");
summaryData = addSummaryData(summaryData, "elevationLoss", 3, "mm");
}
// put("Speed", Arrays.asList( // put("Speed", Arrays.asList(
// "averageSpeed", "maxSpeed", "minSpeed", "averageKMPaceSeconds", "minPace", // "averageSpeed", "maxSpeed", "minSpeed", "averageKMPaceSeconds", "minPace",
// "maxPace", "averageSpeed2", "averageCadence", "maxCadence", "minCadence" // "maxPace", "averageSpeed2", "averageCadence", "maxCadence", "minCadence"
// )); // ));
//summaryData = addSummaryData(summaryData,"averageSpeed",3,"mm"); if (analyticsObject.has("Speed")) {
summaryData = addSummaryData(summaryData,"maxSpeed",3,"mm"); //summaryData = addSummaryData(summaryData,"averageSpeed",averageOfJSONArray(analyticsObject.getJSONArray("Speed")),"mm"); // This seems to be calculated somewhere else automatically.
summaryData = addSummaryData(summaryData,"minSpeed",3,"mm"); summaryData = addSummaryData(summaryData, "maxSpeed", maxOfJSONArray(analyticsObject.getJSONArray("Speed")), "m/s");
summaryData = addSummaryData(summaryData,"averageKMPaceSeconds",3,"mm"); summaryData = addSummaryData(summaryData, "minSpeed", maxOfJSONArray(analyticsObject.getJSONArray("Speed")), "m/s");
summaryData = addSummaryData(summaryData,"minPace",3,"mm"); //summaryData = addSummaryData(summaryData, "averageKMPaceSeconds", averageOfJSONArray(analyticsObject.getJSONArray("Pace")), "s/km"); // Is this also calculated automatically then?
summaryData = addSummaryData(summaryData,"maxPace",3,"mm"); //summaryData = addSummaryData(summaryData, "averageKMPaceSeconds",
summaryData = addSummaryData(summaryData,"averageSpeed2",3,"mm"); // (float) (1000.0 * analyticsObject.getJSONArray("Elapsed Time").getDouble(logLength-1) /
summaryData = addSummaryData(summaryData,"averageCadence",3,"mm"); // analyticsObject.getJSONArray("Total Distance").getDouble(logLength-1)),
summaryData = addSummaryData(summaryData,"maxCadence",3,"mm"); // "s/km"
summaryData = addSummaryData(summaryData,"minCadence",3,"mm"); //);
summaryData = addSummaryData(summaryData, "minPace", minOfJSONArray(analyticsObject.getJSONArray("Pace")), "s/km");
summaryData = addSummaryData(summaryData, "maxPace", maxOfJSONArray(analyticsObject.getJSONArray("Pace")), "s/km");
//summaryData = addSummaryData(summaryData,"averageSpeed2",3,"mm");
}
if (analyticsObject.has("Cadence")) {
summaryData = addSummaryData(summaryData, "averageCadence",
60 * sumOfJSONArray(storedLogObject.getJSONArray("Steps")) /
(float) analyticsObject.getJSONArray("Elapsed Time").getDouble(logLength-1),
"steps/min"
);
summaryData = addSummaryData(summaryData, "maxCadence", maxOfJSONArray(analyticsObject.getJSONArray("Cadence")), "steps/min");
summaryData = addSummaryData(summaryData, "minCadence", minOfJSONArray(analyticsObject.getJSONArray("Cadence")), "steps/min");
}
// put("Activity", Arrays.asList( // put("Activity", Arrays.asList(
// "distanceMeters", "steps", "activeSeconds", "caloriesBurnt", "totalStride", // "distanceMeters", "steps", "activeSeconds", "caloriesBurnt", "totalStride",
// "averageHR", "maxHR", "minHR", "averageStride", "maxStride", "minStride" // "averageHR", "maxHR", "minHR", "averageStride", "maxStride", "minStride"
// )); // ));
summaryData = addSummaryData(summaryData,"distanceMeters",3,"m"); if (analyticsObject.has("Intermediate Distance")) summaryData =
summaryData = addSummaryData(summaryData,"steps",3,"mm"); addSummaryData(summaryData,"distanceMeters",
summaryData = addSummaryData(summaryData,"activeSeconds",3,"mm"); (float) analyticsObject.getJSONArray("Total Distance").getDouble(logLength-1),
summaryData = addSummaryData(summaryData,"caloriesBurnt",3,"mm"); "m");
summaryData = addSummaryData(summaryData,"totalStride",3,"mm"); if (storedLogObject.has("Steps")) summaryData = addSummaryData(summaryData, "steps", sumOfJSONArray(storedLogObject.getJSONArray("Steps")),"steps");
summaryData = addSummaryData(summaryData,"averageHR",3,"mm"); //summaryData = addSummaryData(summaryData,"activeSeconds",3,"mm"); // FIXME: Is this suppose to exclude the time of inactivity in a workout?
summaryData = addSummaryData(summaryData,"maxHR",3,"mm"); //summaryData = addSummaryData(summaryData,"caloriesBurnt",3,"mm"); // TODO: Should this be calculated on Gadgetbridge side or be reported by Bangle.js?
summaryData = addSummaryData(summaryData,"minHR",3,"mm"); //summaryData = addSummaryData(summaryData,"totalStride",3,"mm"); // FIXME: What is this?
if (storedLogObject.has("Latitude") && storedLogObject.has("Steps")) { if (storedLogObject.has("Heartrate")) {
summaryData = addSummaryData(summaryData, "averageStride", 3, "mm"); summaryData = addSummaryData(summaryData, "averageHR", averageOfJSONArray(storedLogObject.getJSONArray("Heartrate")), "bpm");
summaryData = addSummaryData(summaryData, "maxStride", 3, "mm"); summaryData = addSummaryData(summaryData, "maxHR", maxOfJSONArray(storedLogObject.getJSONArray("Heartrate")), "bpm");
summaryData = addSummaryData(summaryData, "minStride", 3, "mm"); summaryData = addSummaryData(summaryData, "minHR", minOfJSONArray(storedLogObject.getJSONArray("Heartrate")), "bpm");
}
if (analyticsObject.has("Stride")) {
summaryData = addSummaryData(summaryData, "averageStride",
2 * (float) analyticsObject.getJSONArray("Total Distance").getDouble(logLength-1) /
sumOfJSONArray(storedLogObject.getJSONArray("Steps")),
"m/stride"); // FIXME: Is this meant to be stride length as I've assumed?
summaryData = addSummaryData(summaryData, "maxStride", maxOfJSONArray(analyticsObject.getJSONArray("Stride")), "m/stride");
summaryData = addSummaryData(summaryData, "minStride", minOfJSONArray(analyticsObject.getJSONArray("Stride")), "m/stride");
} }
// put("HeartRateZones", Arrays.asList( // put("HeartRateZones", Arrays.asList(
// "hrZoneNa", "hrZoneWarmUp", "hrZoneFatBurn", "hrZoneAerobic", "hrZoneAnaerobic", // "hrZoneNa", "hrZoneWarmUp", "hrZoneFatBurn", "hrZoneAerobic", "hrZoneAnaerobic",
// "hrZoneExtreme" // "hrZoneExtreme"
// )); // ));
summaryData = addSummaryData(summaryData,"hrZoneNa",3,"mm"); // TODO: Implement hrZones by doing calculations on Gadgetbridge side or make Bangle.js report this (Karvonen method implemented to a degree in watch app "Run+")?
summaryData = addSummaryData(summaryData,"hrZoneWarmUp",3,"mm"); //summaryData = addSummaryData(summaryData,"hrZoneNa",3,"mm");
summaryData = addSummaryData(summaryData,"hrZoneFatBurn",3,"mm"); //summaryData = addSummaryData(summaryData,"hrZoneWarmUp",3,"mm");
summaryData = addSummaryData(summaryData,"hrZoneAerobic",3,"mm"); //summaryData = addSummaryData(summaryData,"hrZoneFatBurn",3,"mm");
summaryData = addSummaryData(summaryData,"hrZoneAnaerobic",3,"mm"); //summaryData = addSummaryData(summaryData,"hrZoneAerobic",3,"mm");
summaryData = addSummaryData(summaryData,"hrZoneExtreme",3,"mm"); //summaryData = addSummaryData(summaryData,"hrZoneAnaerobic",3,"mm");
//summaryData = addSummaryData(summaryData,"hrZoneExtreme",3,"mm");
// put("TrainingEffect", Arrays.asList( // put("TrainingEffect", Arrays.asList(
// "aerobicTrainingEffect", "anaerobicTrainingEffect", "currentWorkoutLoad", // "aerobicTrainingEffect", "anaerobicTrainingEffect", "currentWorkoutLoad",
// "maximumOxygenUptake" // "maximumOxygenUptake"
@ -995,8 +1058,9 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
// put("Laps", Arrays.asList( // put("Laps", Arrays.asList(
// "averageLapPace", "laps" // "averageLapPace", "laps"
// )); // ));
summaryData = addSummaryData(summaryData,"averageLapPace",3,"mm"); // TODO: Does Bangle.js report laps in recorder logs?
summaryData = addSummaryData(summaryData,"laps",3,"mm"); //summaryData = addSummaryData(summaryData,"averageLapPace",3,"mm");
//summaryData = addSummaryData(summaryData,"laps",3,"mm");
// }}; // }};
summary.setSummaryData(summaryData.toString()); summary.setSummaryData(summaryData.toString());
@ -2297,20 +2361,20 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
return summaryData; return summaryData;
} }
protected JSONObject addSummaryData(JSONObject summaryData, String key, String value) { // protected JSONObject addSummaryData(JSONObject summaryData, String key, String value) {
if (key != null && !key.equals("") && value != null && !value.equals("")) { // if (key != null && !key.equals("") && value != null && !value.equals("")) {
try { // try {
JSONObject innerData = new JSONObject(); // JSONObject innerData = new JSONObject();
innerData.put("value", value); // innerData.put("value", value);
innerData.put("unit", "string"); // innerData.put("unit", "string");
summaryData.put(key, innerData); // summaryData.put(key, innerData);
} catch (JSONException ignore) { // } catch (JSONException ignore) {
} // }
} // }
return summaryData; // return summaryData;
} // }
public String distanceFromCoordinatePairs(String latA, String lonA, String latB, String lonB) { private String distanceFromCoordinatePairs(String latA, String lonA, String latB, String lonB) {
// https://en.wikipedia.org/wiki/Geographic_coordinate_system#Length_of_a_degree // https://en.wikipedia.org/wiki/Geographic_coordinate_system#Length_of_a_degree
//phi = latitude //phi = latitude
//lambda = longitude //lambda = longitude
@ -2331,4 +2395,32 @@ public class BangleJSDeviceSupport extends AbstractBTLEDeviceSupport {
return String.valueOf(sqrt(latDist*latDist+lonDist*lonDist)); return String.valueOf(sqrt(latDist*latDist+lonDist*lonDist));
} }
private float sumOfJSONArray(JSONArray a) throws JSONException {
double sum = 0;
for (int i=0; i<a.length(); i++) {
sum += a.getDouble(i);
}
return (float) sum;
}
private float averageOfJSONArray(JSONArray a) throws JSONException {
return sumOfJSONArray(a) / a.length();
}
private float minOfJSONArray(JSONArray a) throws JSONException {
double min = a.getDouble(0);
for (int i=1; i<a.length(); i++) {
min = Math.min(min, a.getDouble(i));
}
return (float) min;
}
private float maxOfJSONArray(JSONArray a) throws JSONException {
double max = a.getDouble(0);
for (int i=1; i<a.length(); i++) {
max = Math.max(max, a.getDouble(i));
}
return (float) max;
}
} }