mirror of
https://github.com/gpxstudio/gpx.studio.git
synced 2025-09-02 16:52:31 +00:00
elevation profile for all selected files (in order)
This commit is contained in:
@@ -9,6 +9,7 @@ function cloneJSON<T>(obj: T): T {
|
|||||||
|
|
||||||
// An abstract class that groups functions that need to be computed recursively in the GPX file hierarchy
|
// An abstract class that groups functions that need to be computed recursively in the GPX file hierarchy
|
||||||
abstract class GPXTreeElement<T extends GPXTreeElement<any>> {
|
abstract class GPXTreeElement<T extends GPXTreeElement<any>> {
|
||||||
|
statistics: GPXStatistics;
|
||||||
|
|
||||||
abstract isLeaf(): boolean;
|
abstract isLeaf(): boolean;
|
||||||
abstract getChildren(): T[];
|
abstract getChildren(): T[];
|
||||||
@@ -26,8 +27,6 @@ abstract class GPXTreeElement<T extends GPXTreeElement<any>> {
|
|||||||
|
|
||||||
// An abstract class that can be extended to facilitate functions working similarly with Tracks and TrackSegments
|
// An abstract class that can be extended to facilitate functions working similarly with Tracks and TrackSegments
|
||||||
abstract class GPXTreeNode<T extends GPXTreeElement<any>> extends GPXTreeElement<T> {
|
abstract class GPXTreeNode<T extends GPXTreeElement<any>> extends GPXTreeElement<T> {
|
||||||
statistics: GPXStatistics;
|
|
||||||
|
|
||||||
isLeaf(): boolean {
|
isLeaf(): boolean {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -86,16 +85,21 @@ abstract class GPXTreeNode<T extends GPXTreeElement<any>> extends GPXTreeElement
|
|||||||
slope: [],
|
slope: [],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let current = new GPXStatistics();
|
||||||
for (let child of this.getChildren()) {
|
for (let child of this.getChildren()) {
|
||||||
let childData = child.getTrackPointsAndStatistics();
|
let childData = child.getTrackPointsAndStatistics();
|
||||||
points = points.concat(childData.points);
|
points = points.concat(childData.points);
|
||||||
statistics.distance = statistics.distance.concat(childData.statistics.distance);
|
|
||||||
statistics.time = statistics.time.concat(childData.statistics.time);
|
statistics.distance = statistics.distance.concat(childData.statistics.distance.map((distance) => distance + current.distance.total));
|
||||||
|
statistics.time = statistics.time.concat(childData.statistics.time.map((time) => time + current.time.total));
|
||||||
|
statistics.elevation.gain = statistics.elevation.gain.concat(childData.statistics.elevation.gain.map((gain) => gain + current.elevation.gain));
|
||||||
|
statistics.elevation.loss = statistics.elevation.loss.concat(childData.statistics.elevation.loss.map((loss) => loss + current.elevation.loss));
|
||||||
|
|
||||||
statistics.speed = statistics.speed.concat(childData.statistics.speed);
|
statistics.speed = statistics.speed.concat(childData.statistics.speed);
|
||||||
statistics.elevation.smoothed = statistics.elevation.smoothed.concat(childData.statistics.elevation.smoothed);
|
statistics.elevation.smoothed = statistics.elevation.smoothed.concat(childData.statistics.elevation.smoothed);
|
||||||
statistics.elevation.gain = statistics.elevation.gain.concat(childData.statistics.elevation.gain);
|
|
||||||
statistics.elevation.loss = statistics.elevation.loss.concat(childData.statistics.elevation.loss);
|
|
||||||
statistics.slope = statistics.slope.concat(childData.statistics.slope);
|
statistics.slope = statistics.slope.concat(childData.statistics.slope);
|
||||||
|
|
||||||
|
current.mergeWith(child.statistics);
|
||||||
}
|
}
|
||||||
|
|
||||||
return { points, statistics };
|
return { points, statistics };
|
||||||
@@ -113,6 +117,26 @@ abstract class GPXTreeLeaf extends GPXTreeElement<GPXTreeLeaf> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// A class that represents a set of GPX files
|
||||||
|
export class GPXFiles extends GPXTreeNode<GPXFile> {
|
||||||
|
files: GPXFile[];
|
||||||
|
|
||||||
|
constructor(files: GPXFile[]) {
|
||||||
|
super();
|
||||||
|
this.files = files;
|
||||||
|
|
||||||
|
this.computeStatistics();
|
||||||
|
}
|
||||||
|
|
||||||
|
getChildren(): GPXFile[] {
|
||||||
|
return this.files;
|
||||||
|
}
|
||||||
|
|
||||||
|
toGeoJSON(): any {
|
||||||
|
return this.getChildren().map((child) => child.toGeoJSON());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// A class that represents a GPX file
|
// A class that represents a GPX file
|
||||||
export class GPXFile extends GPXTreeNode<Track>{
|
export class GPXFile extends GPXTreeNode<Track>{
|
||||||
attributes: GPXFileAttributes;
|
attributes: GPXFileAttributes;
|
||||||
@@ -305,6 +329,7 @@ export class TrackSegment extends GPXTreeLeaf {
|
|||||||
|
|
||||||
trkptStatistics.speed = distanceWindowSmoothingWithDistanceAccumulator(points, 200, (accumulated, start, end) => 3600 * accumulated / (points[end].time.getTime() - points[start].time.getTime()));
|
trkptStatistics.speed = distanceWindowSmoothingWithDistanceAccumulator(points, 200, (accumulated, start, end) => 3600 * accumulated / (points[end].time.getTime() - points[start].time.getTime()));
|
||||||
|
|
||||||
|
this.statistics = statistics;
|
||||||
this.trkptStatistics = trkptStatistics;
|
this.trkptStatistics = trkptStatistics;
|
||||||
|
|
||||||
return statistics;
|
return statistics;
|
||||||
|
@@ -5,7 +5,7 @@
|
|||||||
|
|
||||||
import Chart from 'chart.js/auto';
|
import Chart from 'chart.js/auto';
|
||||||
|
|
||||||
import { selectedFiles } from '$lib/stores';
|
import { files, fileOrder, selectedFiles } from '$lib/stores';
|
||||||
|
|
||||||
import { onDestroy, onMount } from 'svelte';
|
import { onDestroy, onMount } from 'svelte';
|
||||||
import {
|
import {
|
||||||
@@ -17,6 +17,7 @@
|
|||||||
Thermometer,
|
Thermometer,
|
||||||
Zap
|
Zap
|
||||||
} from 'lucide-svelte';
|
} from 'lucide-svelte';
|
||||||
|
import { GPXFiles } from 'gpx';
|
||||||
|
|
||||||
let canvas: HTMLCanvasElement;
|
let canvas: HTMLCanvasElement;
|
||||||
let chart: Chart;
|
let chart: Chart;
|
||||||
@@ -125,10 +126,14 @@
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
$: {
|
$: if (chart) {
|
||||||
if ($selectedFiles.size == 1) {
|
let gpxFiles = new GPXFiles(Array.from($selectedFiles));
|
||||||
$selectedFiles.forEach((file) => {
|
let order = $fileOrder.length == 0 ? $files : $fileOrder;
|
||||||
const trackPointsAndStatistics = file.getTrackPointsAndStatistics();
|
gpxFiles.files.sort(function (a, b) {
|
||||||
|
return order.indexOf(a) - order.indexOf(b);
|
||||||
|
});
|
||||||
|
|
||||||
|
let trackPointsAndStatistics = gpxFiles.getTrackPointsAndStatistics();
|
||||||
chart.data.datasets[0] = {
|
chart.data.datasets[0] = {
|
||||||
label: 'Elevation',
|
label: 'Elevation',
|
||||||
data: trackPointsAndStatistics.points.map((point, index) => {
|
data: trackPointsAndStatistics.points.map((point, index) => {
|
||||||
@@ -201,11 +206,9 @@
|
|||||||
hidden: true
|
hidden: true
|
||||||
};
|
};
|
||||||
chart.options.scales.x['min'] = 0;
|
chart.options.scales.x['min'] = 0;
|
||||||
chart.options.scales.x['max'] = file.statistics.distance.total;
|
chart.options.scales.x['max'] = gpxFiles.statistics.distance.total;
|
||||||
});
|
|
||||||
chart.update();
|
chart.update();
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
$: console.log(elevationFill);
|
$: console.log(elevationFill);
|
||||||
$: if (additionalDatasets && chart) {
|
$: if (additionalDatasets && chart) {
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { files, selectedFiles, selectFiles } from '$lib/stores';
|
import { fileOrder, files, selectedFiles, selectFiles } from '$lib/stores';
|
||||||
|
|
||||||
import { ScrollArea } from '$lib/components/ui/scroll-area/index';
|
import { ScrollArea } from '$lib/components/ui/scroll-area/index';
|
||||||
|
|
||||||
@@ -57,6 +57,9 @@
|
|||||||
onDeselect: (e) => {
|
onDeselect: (e) => {
|
||||||
const index = parseInt(e.item.getAttribute('data-id'));
|
const index = parseInt(e.item.getAttribute('data-id'));
|
||||||
deselectFile($files[index]);
|
deselectFile($files[index]);
|
||||||
|
},
|
||||||
|
onSort: () => {
|
||||||
|
$fileOrder = sortable.toArray().map((index) => $files[parseInt(index)]);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@@ -5,6 +5,7 @@ import { GPXFile, buildGPX, parseGPX } from 'gpx';
|
|||||||
|
|
||||||
export const map = writable<mapboxgl.Map | null>(null);
|
export const map = writable<mapboxgl.Map | null>(null);
|
||||||
export const files = writable<GPXFile[]>([]);
|
export const files = writable<GPXFile[]>([]);
|
||||||
|
export const fileOrder = writable<GPXFile[]>([]);
|
||||||
export const selectedFiles = writable<Set<GPXFile>>(new Set());
|
export const selectedFiles = writable<Set<GPXFile>>(new Set());
|
||||||
export const selectFiles = writable<{ [key: string]: (file: GPXFile) => void }>({});
|
export const selectFiles = writable<{ [key: string]: (file: GPXFile) => void }>({});
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user