Files
gpx.studio/gpx/src/io.ts

181 lines
6.3 KiB
TypeScript
Raw Normal View History

import { XMLParser, XMLBuilder } from 'fast-xml-parser';
import { GPXFileType } from './types';
import { GPXFile } from './gpx';
2024-04-15 14:26:34 +02:00
2025-01-09 20:45:11 +01:00
const attributesWithNamespace = {
RoutePointExtension: 'gpxx:RoutePointExtension',
rpt: 'gpxx:rpt',
TrackPointExtension: 'gpxtpx:TrackPointExtension',
PowerExtension: 'gpxpx:PowerExtension',
atemp: 'gpxtpx:atemp',
hr: 'gpxtpx:hr',
cad: 'gpxtpx:cad',
Extensions: 'gpxtpx:Extensions',
PowerInWatts: 'gpxpx:PowerInWatts',
power: 'gpxpx:PowerExtension',
line: 'gpx_style:line',
color: 'gpx_style:color',
opacity: 'gpx_style:opacity',
width: 'gpx_style:width',
2025-01-09 20:45:11 +01:00
};
2025-04-04 08:56:53 +02:00
const floatPatterns = [
/[-+]?\d*\.\d+$/, // decimal
/[-+]?\d+$/, // integer
];
function safeParseFloat(value: string): number {
const parsed = parseFloat(value);
if (!isNaN(parsed)) {
return parsed;
}
for (const pattern of floatPatterns) {
const match = value.match(pattern);
if (match) {
return parseFloat(match[0]);
}
}
return 0.0;
}
2024-04-15 14:26:34 +02:00
export function parseGPX(gpxData: string): GPXFile {
const parser = new XMLParser({
ignoreAttributes: false,
attributeNamePrefix: '',
2024-04-16 13:02:22 +02:00
attributesGroupName: 'attributes',
2025-01-09 20:45:11 +01:00
removeNSPrefix: true,
isArray(name: string) {
return (
name === 'trk' ||
name === 'trkseg' ||
name === 'trkpt' ||
name === 'wpt' ||
name === 'rte' ||
name === 'rtept' ||
name === 'gpxx:rpt'
);
},
attributeValueProcessor(attrName, attrValue, jPath) {
if (attrName === 'lat' || attrName === 'lon') {
2025-04-04 08:56:53 +02:00
return safeParseFloat(attrValue);
}
return attrValue;
},
transformTagName(tagName: string) {
2025-01-09 20:45:11 +01:00
if (attributesWithNamespace[tagName]) {
return attributesWithNamespace[tagName];
}
return tagName;
},
2024-07-22 18:27:53 +02:00
parseTagValue: false,
tagValueProcessor(tagName, tagValue, jPath, hasAttributes, isLeafNode) {
if (isLeafNode) {
if (tagName === 'ele') {
2025-04-04 08:56:53 +02:00
return safeParseFloat(tagValue);
}
if (tagName === 'time') {
return new Date(tagValue);
}
if (
tagName === 'gpxtpx:atemp' ||
tagName === 'gpxtpx:hr' ||
tagName === 'gpxtpx:cad' ||
tagName === 'gpxpx:PowerInWatts' ||
tagName === 'gpx_style:opacity' ||
tagName === 'gpx_style:width'
) {
2025-04-04 08:56:53 +02:00
return safeParseFloat(tagValue);
}
2024-04-16 13:02:22 +02:00
if (tagName === 'gpxpx:PowerExtension') {
// Finish the transformation of the simple <power> tag to the more complex <gpxpx:PowerExtension> tag
// Note that this only targets the transformed <power> tag, since it must be a leaf node
return {
2025-04-04 08:56:53 +02:00
'gpxpx:PowerInWatts': safeParseFloat(tagValue),
};
}
2024-04-16 09:54:41 +02:00
}
return tagValue;
},
});
2024-04-16 09:54:41 +02:00
2024-04-16 13:54:48 +02:00
const parsed: GPXFileType = parser.parse(gpxData).gpx;
2024-04-16 09:54:41 +02:00
// @ts-ignore
if (parsed.metadata === '') {
parsed.metadata = {};
}
2024-04-16 13:54:48 +02:00
return new GPXFile(parsed);
}
2024-04-16 13:02:22 +02:00
2024-07-23 11:20:31 +02:00
export function buildGPX(file: GPXFile, exclude: string[]): string {
2024-09-10 20:17:33 +02:00
const gpx = file.toGPXFileType(exclude);
let lastDate = undefined;
2024-04-16 13:02:22 +02:00
const builder = new XMLBuilder({
format: true,
ignoreAttributes: false,
attributeNamePrefix: '',
2024-04-16 13:02:22 +02:00
attributesGroupName: 'attributes',
suppressEmptyNode: true,
tagValueProcessor: (tagName: string, tagValue: unknown): string | undefined => {
2024-04-16 13:02:22 +02:00
if (tagValue instanceof Date) {
if (isNaN(tagValue.getTime())) {
return lastDate?.toISOString();
}
lastDate = tagValue;
2024-04-16 13:02:22 +02:00
return tagValue.toISOString();
}
return tagValue.toString();
},
});
if (!gpx.attributes) gpx.attributes = {};
gpx.attributes['creator'] = gpx.attributes['creator'] ?? 'https://gpx.studio';
2024-04-16 13:02:22 +02:00
gpx.attributes['version'] = '1.1';
gpx.attributes['xmlns'] = 'http://www.topografix.com/GPX/1/1';
gpx.attributes['xmlns:xsi'] = 'http://www.w3.org/2001/XMLSchema-instance';
gpx.attributes['xsi:schemaLocation'] =
'http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd http://www.garmin.com/xmlschemas/GpxExtensions/v3 http://www.garmin.com/xmlschemas/GpxExtensionsv3.xsd http://www.garmin.com/xmlschemas/TrackPointExtension/v1 http://www.garmin.com/xmlschemas/TrackPointExtensionv1.xsd http://www.garmin.com/xmlschemas/PowerExtension/v1 http://www.garmin.com/xmlschemas/PowerExtensionv1.xsd http://www.topografix.com/GPX/gpx_style/0/2 http://www.topografix.com/GPX/gpx_style/0/2/gpx_style.xsd';
2024-04-16 13:02:22 +02:00
gpx.attributes['xmlns:gpxtpx'] = 'http://www.garmin.com/xmlschemas/TrackPointExtension/v1';
gpx.attributes['xmlns:gpxx'] = 'http://www.garmin.com/xmlschemas/GpxExtensions/v3';
gpx.attributes['xmlns:gpxpx'] = 'http://www.garmin.com/xmlschemas/PowerExtension/v1';
gpx.attributes['xmlns:gpx_style'] = 'http://www.topografix.com/GPX/gpx_style/0/2';
if (gpx.trk.length === 1 && (gpx.trk[0].name === undefined || gpx.trk[0].name === '')) {
gpx.trk[0].name = gpx.metadata.name;
}
2024-04-16 13:02:22 +02:00
return builder.build({
'?xml': {
2024-04-16 13:02:22 +02:00
attributes: {
version: '1.0',
encoding: 'UTF-8',
},
2024-04-16 13:02:22 +02:00
},
gpx: removeEmptyElements(gpx),
2024-04-16 13:02:22 +02:00
});
2024-09-10 16:45:37 +02:00
}
function removeEmptyElements(obj: GPXFileType): GPXFileType {
for (const key in obj) {
if (
obj[key] === null ||
obj[key] === undefined ||
obj[key] === '' ||
(Array.isArray(obj[key]) && obj[key].length === 0)
) {
2024-09-10 16:45:37 +02:00
delete obj[key];
2024-09-11 15:08:44 +02:00
} else if (typeof obj[key] === 'object' && !(obj[key] instanceof Date)) {
2024-09-10 16:45:37 +02:00
removeEmptyElements(obj[key]);
if (Object.keys(obj[key]).length === 0) {
delete obj[key];
}
}
}
return obj;
}