Moved loading code to Task (not working)
This commit is contained in:
parent
74cc5a6579
commit
2715b66d5b
8 changed files with 224 additions and 135 deletions
|
@ -24,7 +24,6 @@
|
||||||
|
|
||||||
package es.kauron.estraba;
|
package es.kauron.estraba;
|
||||||
|
|
||||||
import es.kauron.estraba.controller.DashboardController;
|
|
||||||
import javafx.application.Application;
|
import javafx.application.Application;
|
||||||
import javafx.fxml.FXMLLoader;
|
import javafx.fxml.FXMLLoader;
|
||||||
import javafx.scene.Parent;
|
import javafx.scene.Parent;
|
||||||
|
@ -51,7 +50,7 @@ public class App extends Application {
|
||||||
@Override
|
@Override
|
||||||
public void start(Stage stage) throws Exception {
|
public void start(Stage stage) throws Exception {
|
||||||
FXMLLoader loader = new FXMLLoader(
|
FXMLLoader loader = new FXMLLoader(
|
||||||
App.class.getResource("fxml/Dashboard.fxml"), GENERAL_BUNDLE);
|
App.class.getResource("fxml/Splash.fxml"), GENERAL_BUNDLE);
|
||||||
Parent root = loader.load();
|
Parent root = loader.load();
|
||||||
|
|
||||||
stage.getIcons().add(new Image(App.class.getResource("img/icon.png").toString()));
|
stage.getIcons().add(new Image(App.class.getResource("img/icon.png").toString()));
|
||||||
|
@ -59,7 +58,11 @@ public class App extends Application {
|
||||||
stage.setResizable(false);
|
stage.setResizable(false);
|
||||||
stage.setScene(new Scene(root));
|
stage.setScene(new Scene(root));
|
||||||
|
|
||||||
|
// Begin awesomewm code
|
||||||
|
stage.setMinHeight(500);
|
||||||
|
stage.setMinWidth(300);
|
||||||
|
stage.setResizable(false);
|
||||||
|
// End awesomewm code
|
||||||
stage.show();
|
stage.show();
|
||||||
loader.<DashboardController>getController().postinit();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,12 +2,14 @@ package es.kauron.estraba.controller;
|
||||||
|
|
||||||
import com.jfoenix.controls.JFXButton;
|
import com.jfoenix.controls.JFXButton;
|
||||||
import com.jfoenix.controls.JFXSnackbar;
|
import com.jfoenix.controls.JFXSnackbar;
|
||||||
|
import com.jfoenix.controls.JFXSpinner;
|
||||||
import com.lynden.gmapsfx.GoogleMapView;
|
import com.lynden.gmapsfx.GoogleMapView;
|
||||||
import com.lynden.gmapsfx.MapComponentInitializedListener;
|
import com.lynden.gmapsfx.MapComponentInitializedListener;
|
||||||
import com.lynden.gmapsfx.javascript.object.*;
|
import com.lynden.gmapsfx.javascript.object.*;
|
||||||
import com.lynden.gmapsfx.shapes.Polyline;
|
import com.lynden.gmapsfx.shapes.Polyline;
|
||||||
import com.lynden.gmapsfx.shapes.PolylineOptions;
|
import com.lynden.gmapsfx.shapes.PolylineOptions;
|
||||||
import es.kauron.estraba.App;
|
import es.kauron.estraba.App;
|
||||||
|
import es.kauron.estraba.model.DataBundle;
|
||||||
import javafx.collections.ObservableList;
|
import javafx.collections.ObservableList;
|
||||||
import javafx.event.ActionEvent;
|
import javafx.event.ActionEvent;
|
||||||
import javafx.fxml.FXML;
|
import javafx.fxml.FXML;
|
||||||
|
@ -15,28 +17,14 @@ import javafx.fxml.Initializable;
|
||||||
import javafx.scene.chart.AreaChart;
|
import javafx.scene.chart.AreaChart;
|
||||||
import javafx.scene.chart.LineChart;
|
import javafx.scene.chart.LineChart;
|
||||||
import javafx.scene.chart.PieChart;
|
import javafx.scene.chart.PieChart;
|
||||||
import javafx.scene.chart.XYChart;
|
|
||||||
import javafx.scene.control.Label;
|
import javafx.scene.control.Label;
|
||||||
import javafx.scene.control.Tab;
|
import javafx.scene.control.Tab;
|
||||||
import javafx.scene.image.Image;
|
import javafx.scene.image.Image;
|
||||||
import javafx.scene.image.ImageView;
|
import javafx.scene.image.ImageView;
|
||||||
import javafx.scene.layout.AnchorPane;
|
import javafx.scene.layout.AnchorPane;
|
||||||
import javafx.stage.FileChooser;
|
|
||||||
import jgpx.model.analysis.Chunk;
|
import jgpx.model.analysis.Chunk;
|
||||||
import jgpx.model.analysis.TrackData;
|
|
||||||
import jgpx.model.gpx.Track;
|
|
||||||
import jgpx.model.jaxb.GpxType;
|
|
||||||
import jgpx.model.jaxb.TrackPointExtensionT;
|
|
||||||
|
|
||||||
import javax.xml.bind.JAXBContext;
|
|
||||||
import javax.xml.bind.JAXBElement;
|
|
||||||
import javax.xml.bind.JAXBException;
|
|
||||||
import javax.xml.bind.Unmarshaller;
|
|
||||||
import java.io.File;
|
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.time.LocalTime;
|
|
||||||
import java.time.format.DateTimeFormatter;
|
|
||||||
import java.time.format.FormatStyle;
|
|
||||||
import java.util.ResourceBundle;
|
import java.util.ResourceBundle;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -59,13 +47,15 @@ public class DashboardController implements Initializable, MapComponentInitializ
|
||||||
private Label valueHRAvg, valueHRMin, valueHRMax, valueSpeedAvg, valueSpeedMax, valueCadenceAvg, valueCadenceMax,
|
private Label valueHRAvg, valueHRMin, valueHRMax, valueSpeedAvg, valueSpeedMax, valueCadenceAvg, valueCadenceMax,
|
||||||
valueDate, valueTime, valueActiveTime, valueTotalTime, valueDistance, valueElevation, labelMotivationUpper,
|
valueDate, valueTime, valueActiveTime, valueTotalTime, valueDistance, valueElevation, labelMotivationUpper,
|
||||||
valueAscent, valueDescent, labelMotivatorLower;
|
valueAscent, valueDescent, labelMotivatorLower;
|
||||||
|
@FXML
|
||||||
|
private JFXSpinner mapSpinner;
|
||||||
|
|
||||||
@FXML
|
@FXML
|
||||||
private PieChart zoneChart;
|
private PieChart zoneChart;
|
||||||
|
|
||||||
@FXML
|
@FXML
|
||||||
private GoogleMapView mapView;
|
private GoogleMapView mapView;
|
||||||
private TrackData track;
|
private ObservableList<Chunk> chunks;
|
||||||
|
|
||||||
@FXML
|
@FXML
|
||||||
private JFXButton elevationButton, speedButton, hrButton, cadenceButton;
|
private JFXButton elevationButton, speedButton, hrButton, cadenceButton;
|
||||||
|
@ -77,8 +67,6 @@ public class DashboardController implements Initializable, MapComponentInitializ
|
||||||
private LineChart<Double, Double> speedChart, hrChart, cadenceChart, mapChart;
|
private LineChart<Double, Double> speedChart, hrChart, cadenceChart, mapChart;
|
||||||
|
|
||||||
private JFXSnackbar snackbar;
|
private JFXSnackbar snackbar;
|
||||||
private static final double DISTANCE_EPSILON = 1E-6;
|
|
||||||
private static final double KILOMETER_CUTOFF = 10000;
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void initialize(URL location, ResourceBundle resources) {
|
public void initialize(URL location, ResourceBundle resources) {
|
||||||
|
@ -117,123 +105,41 @@ public class DashboardController implements Initializable, MapComponentInitializ
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void postinit() {
|
void postInit(DataBundle bundle) {
|
||||||
snackbar.registerSnackbarContainer(root);
|
snackbar.registerSnackbarContainer(root);
|
||||||
try {load();} catch (JAXBException e) {e.printStackTrace();}
|
loadTrack(bundle);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void loadTrack(TrackData track) {
|
private void loadTrack(DataBundle bundle) {
|
||||||
valueHRAvg.setText(track.getAverageHeartrate()
|
valueHRAvg.setText(bundle.HRAvg);
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.bpm"));
|
valueHRMax.setText(bundle.HRMax);
|
||||||
valueHRMax.setText(track.getMaxHeartrate()
|
valueHRMin.setText(bundle.HRMin);
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.bpm"));
|
valueSpeedAvg.setText(bundle.speedAvg);
|
||||||
valueHRMin.setText(track.getMinHeartRate()
|
valueSpeedMax.setText(bundle.speedMax);
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.bpm"));
|
valueCadenceAvg.setText(bundle.cadenceAvg);
|
||||||
|
valueCadenceMax.setText(bundle.cadenceMax);
|
||||||
|
valueDate.setText(bundle.date);
|
||||||
|
valueTime.setText(bundle.time);
|
||||||
|
valueActiveTime.setText(bundle.activeTime);
|
||||||
|
valueTotalTime.setText(bundle.totalTime);
|
||||||
|
valueDistance.setText(bundle.distance);
|
||||||
|
valueElevation.setText(bundle.elevation);
|
||||||
|
valueAscent.setText(bundle.ascent);
|
||||||
|
valueDescent.setText(bundle.descent);
|
||||||
|
|
||||||
// speed is given as m/s
|
zoneChart.setData(bundle.pieData);
|
||||||
valueSpeedAvg.setText(String.format("%.2f", track.getAverageSpeed() * 3.6)
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.kmph"));
|
|
||||||
valueSpeedMax.setText(String.format("%.2f", track.getMaxSpeed() * 3.6)
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.kmph"));
|
|
||||||
|
|
||||||
valueCadenceAvg.setText(track.getAverageCadence()
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.hz"));
|
|
||||||
valueCadenceMax.setText(track.getMaxCadence()
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.hz"));
|
|
||||||
|
|
||||||
valueDate.setText(track.getStartTime().format(DateTimeFormatter.ofLocalizedDate(FormatStyle.FULL)));
|
|
||||||
valueTime.setText(track.getStartTime().format(DateTimeFormatter.ofLocalizedTime(FormatStyle.MEDIUM)));
|
|
||||||
valueActiveTime.setText(LocalTime.MIDNIGHT.plus(track.getMovingTime())
|
|
||||||
.format(DateTimeFormatter.ofPattern("HH:mm:ss")));
|
|
||||||
valueTotalTime.setText(App.GENERAL_BUNDLE.getString("time.of")
|
|
||||||
+ LocalTime.MIDNIGHT.plus(track.getTotalDuration())
|
|
||||||
.format(DateTimeFormatter.ofPattern("HH:mm:ss")));
|
|
||||||
|
|
||||||
if (track.getTotalDistance() > KILOMETER_CUTOFF) {
|
|
||||||
valueDistance.setText(String.format("%.2f", track.getTotalDistance() / 1000)
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.km"));
|
|
||||||
} else {
|
|
||||||
valueDistance.setText(String.format("%.2f", track.getTotalDistance())
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.m"));
|
|
||||||
}
|
|
||||||
|
|
||||||
valueElevation.setText((int)(track.getTotalAscent() - track.getTotalDescend())
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.m"));
|
|
||||||
valueAscent.setText((int)track.getTotalAscent()
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.m"));
|
|
||||||
valueDescent.setText((int)track.getTotalDescend()
|
|
||||||
+ App.GENERAL_BUNDLE.getString("unit.m"));
|
|
||||||
|
|
||||||
// create charts data
|
|
||||||
XYChart.Series<Double, Double> elevationChartData = new XYChart.Series<>();
|
|
||||||
XYChart.Series<Double, Double> speedChartData = new XYChart.Series<>();
|
|
||||||
XYChart.Series<Double, Double> hrChartData = new XYChart.Series<>();
|
|
||||||
XYChart.Series<Double, Double> cadenceChartData = new XYChart.Series<>();
|
|
||||||
|
|
||||||
// traverse the chunks
|
|
||||||
ObservableList<Chunk> chunks = track.getChunks();
|
|
||||||
double currentDistance = 0.0;
|
|
||||||
double currentHeight = chunks.get(0).getFirstPoint().getElevation();
|
|
||||||
for (Chunk chunk : chunks) {
|
|
||||||
currentDistance += chunk.getDistance();
|
|
||||||
if (chunk.getDistance() < DISTANCE_EPSILON) continue;
|
|
||||||
currentHeight += chunk.getAscent() - chunk.getDescend();
|
|
||||||
|
|
||||||
elevationChartData.getData().add(new XYChart.Data<>(currentDistance, currentHeight));
|
|
||||||
speedChartData.getData().add(new XYChart.Data<>(currentDistance, chunk.getSpeed()*3.6)); // m/s
|
|
||||||
hrChartData.getData().add(new XYChart.Data<>(currentDistance, chunk.getAvgHeartRate()));
|
|
||||||
cadenceChartData.getData().add(new XYChart.Data<>(currentDistance, chunk.getAvgCadence()));
|
|
||||||
|
|
||||||
String zone;
|
|
||||||
if (chunk.getAvgHeartRate() > 170) zone = App.GENERAL_BUNDLE.getString("zone.anaerobic");
|
|
||||||
else if (chunk.getAvgHeartRate() > 150) zone = App.GENERAL_BUNDLE.getString("zone.threshold");
|
|
||||||
else if (chunk.getAvgHeartRate() > 130) zone = App.GENERAL_BUNDLE.getString("zone.tempo");
|
|
||||||
else if (chunk.getAvgHeartRate() > 110) zone = App.GENERAL_BUNDLE.getString("zone.endurance");
|
|
||||||
else zone = App.GENERAL_BUNDLE.getString("zone.recovery");
|
|
||||||
|
|
||||||
boolean pieFound = false;
|
|
||||||
for (PieChart.Data d : zoneChart.getData()){
|
|
||||||
if (d.getName().equals(zone)) {
|
|
||||||
pieFound = true;
|
|
||||||
d.setPieValue(d.getPieValue() + 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!pieFound) zoneChart.getData().add( new PieChart.Data(zone, 1) );
|
|
||||||
}
|
|
||||||
|
|
||||||
// populate the charts
|
// populate the charts
|
||||||
elevationChart.getData().add(elevationChartData);
|
elevationChart.getData().add(bundle.elevationSeries);
|
||||||
speedChart.getData().add(speedChartData);
|
speedChart.getData().add(bundle.speedSeries);
|
||||||
hrChart.getData().add(hrChartData);
|
hrChart.getData().add(bundle.hrSeries);
|
||||||
cadenceChart.getData().add(cadenceChartData);
|
cadenceChart.getData().add(bundle.cadenceSeries);
|
||||||
|
|
||||||
//initialize map
|
//initialize map
|
||||||
|
chunks = bundle.chunks;
|
||||||
mapView.addMapInializedListener(this);
|
mapView.addMapInializedListener(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void load() throws JAXBException {
|
|
||||||
FileChooser fileChooser = new FileChooser();
|
|
||||||
fileChooser.getExtensionFilters().add(
|
|
||||||
new FileChooser.ExtensionFilter(App.GENERAL_BUNDLE.getString("app.extension.filter.name"), "*.gpx"));
|
|
||||||
File file = fileChooser.showOpenDialog(root.getScene().getWindow());
|
|
||||||
if (file == null) return;
|
|
||||||
|
|
||||||
String name = file.getName();
|
|
||||||
JAXBContext jaxbContext = JAXBContext.newInstance(GpxType.class, TrackPointExtensionT.class);
|
|
||||||
Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
|
|
||||||
@SuppressWarnings("unchecked")
|
|
||||||
JAXBElement<Object> jaxbElement = (JAXBElement<Object>) unmarshaller.unmarshal(file);
|
|
||||||
GpxType gpx = (GpxType) jaxbElement.getValue();
|
|
||||||
|
|
||||||
if (gpx != null) {
|
|
||||||
track = new TrackData(new Track(gpx.getTrk().get(0)));
|
|
||||||
loadTrack(track);
|
|
||||||
snackbar.show("GPX file: " + name + "successfully loaded", 3000);
|
|
||||||
} else {
|
|
||||||
snackbar.show("Error loading GPX file: " + name, 3000);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void mapInitialized() {
|
public void mapInitialized() {
|
||||||
// When the JS init is done
|
// When the JS init is done
|
||||||
|
@ -260,10 +166,10 @@ public class DashboardController implements Initializable, MapComponentInitializ
|
||||||
// Prepare an array with LatLong objects
|
// Prepare an array with LatLong objects
|
||||||
MVCArray pathArray = new MVCArray();
|
MVCArray pathArray = new MVCArray();
|
||||||
pathArray.push(new LatLong( // first step of the route
|
pathArray.push(new LatLong( // first step of the route
|
||||||
track.getChunks().get(0).getFirstPoint().getLatitude(),
|
chunks.get(0).getFirstPoint().getLatitude(),
|
||||||
track.getChunks().get(0).getFirstPoint().getLongitude()
|
chunks.get(0).getFirstPoint().getLongitude()
|
||||||
));
|
));
|
||||||
track.getChunks().forEach(chunk -> {
|
chunks.forEach(chunk -> {
|
||||||
double lat = chunk.getLastPoint().getLatitude();
|
double lat = chunk.getLastPoint().getLatitude();
|
||||||
double lon = chunk.getLastPoint().getLongitude();
|
double lon = chunk.getLastPoint().getLongitude();
|
||||||
coord[N] = Math.max(lat, coord[N]);
|
coord[N] = Math.max(lat, coord[N]);
|
||||||
|
@ -283,9 +189,8 @@ public class DashboardController implements Initializable, MapComponentInitializ
|
||||||
new LatLong(coord[N], coord[E])
|
new LatLong(coord[N], coord[E])
|
||||||
));
|
));
|
||||||
map.setZoom(getBoundsZoomLevel(coord, mapView.getHeight(), mapView.getWidth()));
|
map.setZoom(getBoundsZoomLevel(coord, mapView.getHeight(), mapView.getWidth()));
|
||||||
// Print some debug info
|
mapView.setVisible(true);
|
||||||
System.err.printf("Bound to coords: %.2fN, %.2S, %.2fE, %.2fW\n", coord[N], coord[S], coord[E], coord[W]);
|
mapSpinner.setVisible(false);
|
||||||
System.err.printf("Selected zoom: %d\n", map.getZoom());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -1,15 +1,27 @@
|
||||||
package es.kauron.estraba.controller;
|
package es.kauron.estraba.controller;
|
||||||
|
|
||||||
import com.jfoenix.controls.JFXButton;
|
import com.jfoenix.controls.JFXButton;
|
||||||
|
import com.jfoenix.controls.JFXSnackbar;
|
||||||
import com.jfoenix.controls.JFXSpinner;
|
import com.jfoenix.controls.JFXSpinner;
|
||||||
import es.kauron.estraba.App;
|
import es.kauron.estraba.App;
|
||||||
|
import es.kauron.estraba.model.DataBundle;
|
||||||
|
import javafx.application.Platform;
|
||||||
|
import javafx.concurrent.Task;
|
||||||
import javafx.event.ActionEvent;
|
import javafx.event.ActionEvent;
|
||||||
import javafx.fxml.FXML;
|
import javafx.fxml.FXML;
|
||||||
|
import javafx.fxml.FXMLLoader;
|
||||||
import javafx.fxml.Initializable;
|
import javafx.fxml.Initializable;
|
||||||
|
import javafx.scene.Parent;
|
||||||
|
import javafx.scene.Scene;
|
||||||
import javafx.scene.control.Label;
|
import javafx.scene.control.Label;
|
||||||
import javafx.scene.image.Image;
|
import javafx.scene.image.Image;
|
||||||
import javafx.scene.image.ImageView;
|
import javafx.scene.image.ImageView;
|
||||||
|
import javafx.scene.layout.AnchorPane;
|
||||||
|
import javafx.stage.FileChooser;
|
||||||
|
import javafx.stage.Stage;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.util.ResourceBundle;
|
import java.util.ResourceBundle;
|
||||||
|
|
||||||
|
@ -20,6 +32,9 @@ import java.util.ResourceBundle;
|
||||||
|
|
||||||
public class SplashController implements Initializable{
|
public class SplashController implements Initializable{
|
||||||
|
|
||||||
|
@FXML
|
||||||
|
private AnchorPane root;
|
||||||
|
|
||||||
@FXML
|
@FXML
|
||||||
private ImageView imgLogo;
|
private ImageView imgLogo;
|
||||||
|
|
||||||
|
@ -32,17 +47,70 @@ public class SplashController implements Initializable{
|
||||||
@FXML
|
@FXML
|
||||||
private JFXButton buttonLoad;
|
private JFXButton buttonLoad;
|
||||||
|
|
||||||
|
private JFXSnackbar snackbar;
|
||||||
|
private File file;
|
||||||
|
|
||||||
@FXML
|
@FXML
|
||||||
void loadGPXFile(ActionEvent event) {
|
private void loadGPXFile(ActionEvent event) {
|
||||||
|
|
||||||
buttonLoad.setVisible(false);
|
buttonLoad.setVisible(false);
|
||||||
labelWelcome.setVisible(false);
|
labelWelcome.setVisible(false);
|
||||||
spinner.setVisible(true);
|
spinner.setVisible(true);
|
||||||
|
snackbar.registerSnackbarContainer(root);
|
||||||
|
snackbar.show("Loading file", 5000);
|
||||||
|
FileChooser fileChooser = new FileChooser();
|
||||||
|
fileChooser.getExtensionFilters().add(
|
||||||
|
new FileChooser.ExtensionFilter(App.GENERAL_BUNDLE.getString("app.extension.filter.name"), "*.gpx"));
|
||||||
|
file = fileChooser.showOpenDialog(root.getScene().getWindow());
|
||||||
|
if (file == null) {
|
||||||
|
errorLoading();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Task<Void> task = new Task<Void>() {
|
||||||
|
@Override
|
||||||
|
protected Void call() throws Exception {
|
||||||
|
try {
|
||||||
|
DataBundle bundle = DataBundle.loadFrom(file);
|
||||||
|
FXMLLoader loader = new FXMLLoader(
|
||||||
|
App.class.getResource("fxml/Dashboard.fxml"), App.GENERAL_BUNDLE);
|
||||||
|
Parent root = loader.load();
|
||||||
|
|
||||||
|
Stage stage = new Stage();
|
||||||
|
stage.getIcons().add(new Image(App.class.getResource("img/icon.png").toString()));
|
||||||
|
stage.setTitle(App.GENERAL_BUNDLE.getString("app.title"));
|
||||||
|
stage.setResizable(false);
|
||||||
|
stage.setScene(new Scene(root));
|
||||||
|
|
||||||
|
// Begin awesomewm code
|
||||||
|
stage.setMinHeight(500);
|
||||||
|
stage.setMinWidth(800);
|
||||||
|
stage.setResizable(false);
|
||||||
|
// End awesomewm code
|
||||||
|
stage.show();
|
||||||
|
loader.<DashboardController>getController().postInit(bundle);
|
||||||
|
Platform.runLater(() -> ((Stage) root.getScene().getWindow()).close());
|
||||||
|
} catch (IOException e) {
|
||||||
|
errorLoading();
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
Thread t = new Thread(task);
|
||||||
|
t.setDaemon(true);
|
||||||
|
t.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void initialize(URL location, ResourceBundle resources) {
|
public void initialize(URL location, ResourceBundle resources) {
|
||||||
imgLogo.setImage(new Image(App.class.getResourceAsStream("img/splash.png")));
|
imgLogo.setImage(new Image(App.class.getResourceAsStream("img/strava-transparent.png")));
|
||||||
|
snackbar = new JFXSnackbar();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void errorLoading() {
|
||||||
|
buttonLoad.setVisible(true);
|
||||||
|
labelWelcome.setVisible(true);
|
||||||
|
spinner.setVisible(false);
|
||||||
|
snackbar.show("Error loading file", 3000);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
107
src/main/java/es/kauron/estraba/model/DataBundle.java
Normal file
107
src/main/java/es/kauron/estraba/model/DataBundle.java
Normal file
|
@ -0,0 +1,107 @@
|
||||||
|
package es.kauron.estraba.model;
|
||||||
|
|
||||||
|
import es.kauron.estraba.App;
|
||||||
|
import javafx.collections.FXCollections;
|
||||||
|
import javafx.collections.ObservableList;
|
||||||
|
import javafx.scene.chart.PieChart;
|
||||||
|
import javafx.scene.chart.XYChart;
|
||||||
|
import jgpx.model.analysis.Chunk;
|
||||||
|
import jgpx.model.analysis.TrackData;
|
||||||
|
import jgpx.model.gpx.Track;
|
||||||
|
import jgpx.model.jaxb.GpxType;
|
||||||
|
import jgpx.model.jaxb.TrackPointExtensionT;
|
||||||
|
|
||||||
|
import javax.xml.bind.JAXBContext;
|
||||||
|
import javax.xml.bind.JAXBElement;
|
||||||
|
import javax.xml.bind.Unmarshaller;
|
||||||
|
import java.io.File;
|
||||||
|
import java.time.LocalTime;
|
||||||
|
import java.time.format.DateTimeFormatter;
|
||||||
|
import java.time.format.FormatStyle;
|
||||||
|
|
||||||
|
public class DataBundle {
|
||||||
|
public static final int N = 0, S = 1, E = 2, W = 3;
|
||||||
|
private static final double DISTANCE_EPSILON = 1E-6;
|
||||||
|
private static final double KILOMETER_CUTOFF = 10000;
|
||||||
|
|
||||||
|
public String HRAvg, HRMax, HRMin, speedAvg, speedMax, cadenceAvg, cadenceMax;
|
||||||
|
public String date, time, activeTime, totalTime, distance, elevation, ascent, descent;
|
||||||
|
public XYChart.Series<Double, Double> elevationSeries = new XYChart.Series<>(),
|
||||||
|
speedSeries = new XYChart.Series<>(),
|
||||||
|
hrSeries = new XYChart.Series<>(),
|
||||||
|
cadenceSeries = new XYChart.Series<>();
|
||||||
|
public ObservableList<PieChart.Data> pieData = FXCollections.emptyObservableList();
|
||||||
|
public ObservableList<Chunk> chunks;
|
||||||
|
|
||||||
|
public static DataBundle loadFrom(File file) throws Exception {
|
||||||
|
String name = file.getName();
|
||||||
|
JAXBElement<Object> jaxbElement;
|
||||||
|
JAXBContext jaxbContext = JAXBContext.newInstance(GpxType.class, TrackPointExtensionT.class);
|
||||||
|
Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
|
||||||
|
jaxbElement = (JAXBElement<Object>) unmarshaller.unmarshal(file);
|
||||||
|
GpxType gpx = (GpxType) jaxbElement.getValue();
|
||||||
|
|
||||||
|
if (gpx == null) throw new Exception();
|
||||||
|
return new DataBundle(new TrackData(new Track(gpx.getTrk().get(0))));
|
||||||
|
}
|
||||||
|
|
||||||
|
private DataBundle(TrackData track) {
|
||||||
|
HRAvg = track.getAverageHeartrate() + App.GENERAL_BUNDLE.getString("unit.bpm");
|
||||||
|
HRMax = (track.getMaxHeartrate() + App.GENERAL_BUNDLE.getString("unit.bpm"));
|
||||||
|
HRMin = track.getMinHeartRate() + App.GENERAL_BUNDLE.getString("unit.bpm");
|
||||||
|
|
||||||
|
// speed is given as m/s
|
||||||
|
speedAvg = String.format("%.2f", track.getAverageSpeed() * 3.6) + App.GENERAL_BUNDLE.getString("unit.kmph");
|
||||||
|
speedMax = String.format("%.2f", track.getMaxSpeed() * 3.6) + App.GENERAL_BUNDLE.getString("unit.kmph");
|
||||||
|
|
||||||
|
cadenceAvg = track.getAverageCadence() + App.GENERAL_BUNDLE.getString("unit.hz");
|
||||||
|
cadenceMax = track.getMaxCadence() + App.GENERAL_BUNDLE.getString("unit.hz");
|
||||||
|
|
||||||
|
date = track.getStartTime().format(DateTimeFormatter.ofLocalizedDate(FormatStyle.FULL));
|
||||||
|
time = track.getStartTime().format(DateTimeFormatter.ofLocalizedTime(FormatStyle.MEDIUM));
|
||||||
|
activeTime = LocalTime.MIDNIGHT.plus(track.getMovingTime()).format(DateTimeFormatter.ofPattern("HH:mm:ss"));
|
||||||
|
totalTime = App.GENERAL_BUNDLE.getString("time.of")
|
||||||
|
+ LocalTime.MIDNIGHT.plus(track.getTotalDuration()).format(DateTimeFormatter.ofPattern("HH:mm:ss"));
|
||||||
|
|
||||||
|
if (track.getTotalDistance() > KILOMETER_CUTOFF) {
|
||||||
|
distance = String.format("%.2f", track.getTotalDistance() / 1000) + App.GENERAL_BUNDLE.getString("unit.km");
|
||||||
|
} else {
|
||||||
|
distance = String.format("%.2f", track.getTotalDistance()) + App.GENERAL_BUNDLE.getString("unit.m");
|
||||||
|
}
|
||||||
|
|
||||||
|
elevation = (int)(track.getTotalAscent() - track.getTotalDescend()) + App.GENERAL_BUNDLE.getString("unit.m");
|
||||||
|
ascent = (int)track.getTotalAscent() + App.GENERAL_BUNDLE.getString("unit.m");
|
||||||
|
descent = (int)track.getTotalDescend() + App.GENERAL_BUNDLE.getString("unit.m");
|
||||||
|
|
||||||
|
// traverse the chunks
|
||||||
|
chunks = track.getChunks();
|
||||||
|
double currentDistance = 0.0;
|
||||||
|
double currentHeight = chunks.get(0).getFirstPoint().getElevation();
|
||||||
|
for (Chunk chunk : chunks) {
|
||||||
|
currentDistance += chunk.getDistance();
|
||||||
|
if (chunk.getDistance() < DISTANCE_EPSILON) continue;
|
||||||
|
currentHeight += chunk.getAscent() - chunk.getDescend();
|
||||||
|
|
||||||
|
elevationSeries.getData().add(new XYChart.Data<>(currentDistance, currentHeight));
|
||||||
|
speedSeries.getData().add(new XYChart.Data<>(currentDistance, chunk.getSpeed()*3.6)); // m/s
|
||||||
|
hrSeries.getData().add(new XYChart.Data<>(currentDistance, chunk.getAvgHeartRate()));
|
||||||
|
cadenceSeries.getData().add(new XYChart.Data<>(currentDistance, chunk.getAvgCadence()));
|
||||||
|
|
||||||
|
String zone;
|
||||||
|
if (chunk.getAvgHeartRate() > 170) zone = App.GENERAL_BUNDLE.getString("zone.anaerobic");
|
||||||
|
else if (chunk.getAvgHeartRate() > 150) zone = App.GENERAL_BUNDLE.getString("zone.threshold");
|
||||||
|
else if (chunk.getAvgHeartRate() > 130) zone = App.GENERAL_BUNDLE.getString("zone.tempo");
|
||||||
|
else if (chunk.getAvgHeartRate() > 110) zone = App.GENERAL_BUNDLE.getString("zone.endurance");
|
||||||
|
else zone = App.GENERAL_BUNDLE.getString("zone.recovery");
|
||||||
|
|
||||||
|
boolean pieFound = false;
|
||||||
|
for (PieChart.Data d : pieData){
|
||||||
|
if (d.getName().equals(zone)) {
|
||||||
|
pieFound = true;
|
||||||
|
d.setPieValue(d.getPieValue() + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!pieFound) pieData.add( new PieChart.Data(zone, 1) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -8,7 +8,7 @@
|
||||||
<?import javafx.scene.layout.AnchorPane?>
|
<?import javafx.scene.layout.AnchorPane?>
|
||||||
<?import javafx.scene.layout.StackPane?>
|
<?import javafx.scene.layout.StackPane?>
|
||||||
|
|
||||||
<AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="300.0" prefWidth="300.0" styleClass="background" stylesheets="@../css/palette.css" xmlns="http://javafx.com/javafx/8.0.76-ea" xmlns:fx="http://javafx.com/fxml/1" fx:controller="es.kauron.estraba.controller.SplashController">
|
<AnchorPane fx:id="root" maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="300.0" prefWidth="300.0" styleClass="background" stylesheets="@../css/palette.css" xmlns="http://javafx.com/javafx/8.0.76-ea" xmlns:fx="http://javafx.com/fxml/1" fx:controller="es.kauron.estraba.controller.SplashController">
|
||||||
<ImageView fx:id="imgLogo" fitHeight="150.0" fitWidth="200.0" pickOnBounds="true" preserveRatio="true" AnchorPane.leftAnchor="50.0" AnchorPane.rightAnchor="50.0" AnchorPane.topAnchor="25.0">
|
<ImageView fx:id="imgLogo" fitHeight="150.0" fitWidth="200.0" pickOnBounds="true" preserveRatio="true" AnchorPane.leftAnchor="50.0" AnchorPane.rightAnchor="50.0" AnchorPane.topAnchor="25.0">
|
||||||
</ImageView>
|
</ImageView>
|
||||||
<JFXSpinner fx:id="spinner" visible="false" AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0" />
|
<JFXSpinner fx:id="spinner" visible="false" AnchorPane.bottomAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" AnchorPane.topAnchor="0.0" />
|
||||||
|
|
|
@ -4,8 +4,10 @@ label.cadence=Cadence
|
||||||
label.distance=Distance
|
label.distance=Distance
|
||||||
label.elevation=Elevation
|
label.elevation=Elevation
|
||||||
label.hr=Heart rate
|
label.hr=Heart rate
|
||||||
|
label.loadGPX=Load GPX file
|
||||||
label.motivation=Welcome!
|
label.motivation=Welcome!
|
||||||
label.speed=Speed
|
label.speed=Speed
|
||||||
|
label.welcome=Welcome!
|
||||||
tab.dashboard=Dashboard
|
tab.dashboard=Dashboard
|
||||||
tab.graph=Stats
|
tab.graph=Stats
|
||||||
tab.map=Your Route
|
tab.map=Your Route
|
||||||
|
|
|
@ -4,8 +4,10 @@ label.cadence=Cadencia
|
||||||
label.distance=Distancia
|
label.distance=Distancia
|
||||||
label.elevation=Elevacion
|
label.elevation=Elevacion
|
||||||
label.hr=YOLO
|
label.hr=YOLO
|
||||||
|
label.loadGPX=Obrir arxiu GPX
|
||||||
label.motivation=\u00a1Benvinguts!
|
label.motivation=\u00a1Benvinguts!
|
||||||
label.speed=Speed
|
label.speed=Speed
|
||||||
|
label.welcome=Benvingut!
|
||||||
tab.dashboard=Sumari
|
tab.dashboard=Sumari
|
||||||
tab.graph=Estad\u00edstiques
|
tab.graph=Estad\u00edstiques
|
||||||
tab.map=La teva ruta
|
tab.map=La teva ruta
|
||||||
|
|
|
@ -4,8 +4,10 @@ label.cadence=Cadencia
|
||||||
label.distance=Distancia
|
label.distance=Distancia
|
||||||
label.elevation=Elevacion
|
label.elevation=Elevacion
|
||||||
label.hr=Pulsaci\u00f3nes
|
label.hr=Pulsaci\u00f3nes
|
||||||
|
label.loadGPX=Abrir archivo GPX
|
||||||
label.motivation=\u00a1Bienvenido!
|
label.motivation=\u00a1Bienvenido!
|
||||||
label.speed=Velocidad
|
label.speed=Velocidad
|
||||||
|
label.welcome=¡Bienvenido!
|
||||||
tab.dashboard=Res\u00famen
|
tab.dashboard=Res\u00famen
|
||||||
tab.graph=Estad\u00edsticas
|
tab.graph=Estad\u00edsticas
|
||||||
tab.map=Tu ruta
|
tab.map=Tu ruta
|
||||||
|
|
Reference in a new issue