1
0
mirror of https://github.com/tumic0/GPXSee.git synced 2024-10-07 07:13:21 +02:00
GPXSee/src/kmlparser.cpp

517 lines
11 KiB
C++
Raw Normal View History

#include "kmlparser.h"
2016-10-27 22:33:35 +02:00
qreal KMLParser::number()
{
bool res;
qreal ret = _reader.readElementText().toDouble(&res);
if (!res)
2016-10-29 12:22:28 +02:00
_reader.raiseError(QString("Invalid %1").arg(
2016-10-27 22:33:35 +02:00
_reader.name().toString()));
return ret;
}
QDateTime KMLParser::time()
{
QDateTime d = QDateTime::fromString(_reader.readElementText(),
Qt::ISODate);
if (!d.isValid())
2016-10-29 12:22:28 +02:00
_reader.raiseError(QString("Invalid %1").arg(
2016-10-27 22:33:35 +02:00
_reader.name().toString()));
return d;
}
2016-10-28 19:12:40 +02:00
bool KMLParser::coord(Trackpoint &trackpoint)
{
QString data = _reader.readElementText();
const QChar *sp, *ep, *cp, *vp;
int c = 0;
qreal val[3];
bool res;
sp = data.constData();
ep = sp + data.size();
for (cp = sp; cp < ep; cp++)
if (!cp->isSpace())
break;
for (vp = cp; cp <= ep; cp++) {
if (cp->isSpace() || cp->isNull()) {
if (c > 2)
return false;
#if QT_VERSION < QT_VERSION_CHECK(5, 0, 0)
val[c] = QString(vp, cp - vp).toDouble(&res);
#else // QT_VERSION < 5
val[c] = QStringRef(&data, vp - sp, cp - vp).toDouble(&res);
#endif // QT_VERSION < 5
if (!res)
return false;
if (c == 1) {
trackpoint.setCoordinates(Coordinates(val[0], val[1]));
if (!trackpoint.coordinates().isValid())
return false;
} else if (c == 2)
trackpoint.setElevation(val[2]);
while (cp->isSpace())
cp++;
vp = cp;
c++;
}
}
return true;
}
2016-10-27 22:33:35 +02:00
bool KMLParser::pointCoordinates(Waypoint &waypoint)
{
QString data = _reader.readElementText();
const QChar *sp, *ep, *cp, *vp;
int c = 0;
qreal val[3];
bool res;
sp = data.constData();
ep = sp + data.size();
2016-10-25 22:50:11 +02:00
for (cp = sp; cp < ep; cp++)
if (!cp->isSpace())
break;
for (vp = cp; cp <= ep; cp++) {
if (*cp == ',') {
2016-11-11 22:04:26 +01:00
if (c > 1)
return false;
#if QT_VERSION < QT_VERSION_CHECK(5, 0, 0)
val[c] = QString(vp, cp - vp).toDouble(&res);
#else // QT_VERSION < 5
val[c] = QStringRef(&data, vp - sp, cp - vp).toDouble(&res);
#endif // QT_VERSION < 5
if (!res)
return false;
c++;
vp = cp + 1;
2016-10-25 22:50:11 +02:00
} else if (cp->isSpace() || cp->isNull()) {
if (c < 1)
return false;
#if QT_VERSION < QT_VERSION_CHECK(5, 0, 0)
val[c] = QString(vp, cp - vp).toDouble(&res);
#else // QT_VERSION < 5
val[c] = QStringRef(&data, vp - sp, cp - vp).toDouble(&res);
#endif // QT_VERSION < 5
if (!res)
return false;
2016-10-27 22:33:35 +02:00
waypoint.setCoordinates(Coordinates(val[0], val[1]));
2016-10-28 19:12:40 +02:00
if (!waypoint.coordinates().isValid())
return false;
2016-10-25 22:50:11 +02:00
if (c == 2)
2016-10-27 22:33:35 +02:00
waypoint.setElevation(val[2]);
2016-10-25 22:50:11 +02:00
while (cp->isSpace())
cp++;
c = 3;
}
}
return true;
}
bool KMLParser::lineCoordinates(TrackData &track)
{
QString data = _reader.readElementText();
const QChar *sp, *ep, *cp, *vp;
int c = 0;
qreal val[3];
bool res;
sp = data.constData();
ep = sp + data.size();
2016-10-25 22:50:11 +02:00
for (cp = sp; cp < ep; cp++)
if (!cp->isSpace())
break;
for (vp = cp; cp <= ep; cp++) {
if (*cp == ',') {
2016-11-11 22:04:26 +01:00
if (c > 1)
return false;
#if QT_VERSION < QT_VERSION_CHECK(5, 0, 0)
val[c] = QString(vp, cp - vp).toDouble(&res);
#else // QT_VERSION < 5
val[c] = QStringRef(&data, vp - sp, cp - vp).toDouble(&res);
#endif // QT_VERSION < 5
if (!res)
return false;
c++;
vp = cp + 1;
2016-10-25 22:50:11 +02:00
} else if (cp->isSpace() || cp->isNull()) {
if (c < 1 || c > 2)
return false;
2016-10-25 22:50:11 +02:00
#if QT_VERSION < QT_VERSION_CHECK(5, 0, 0)
val[c] = QString(vp, cp - vp).toDouble(&res);
#else // QT_VERSION < 5
val[c] = QStringRef(&data, vp - sp, cp - vp).toDouble(&res);
#endif // QT_VERSION < 5
if (!res)
return false;
track.append(Trackpoint(Coordinates(val[0], val[1])));
2016-10-28 19:12:40 +02:00
if (!track.last().coordinates().isValid())
return false;
2016-10-25 22:50:11 +02:00
if (c == 2)
track.last().setElevation(val[2]);
2016-10-25 22:50:11 +02:00
while (cp->isSpace())
cp++;
c = 0;
2016-10-25 22:50:11 +02:00
vp = cp;
}
}
2016-10-25 22:50:11 +02:00
return true;
}
QDateTime KMLParser::timeStamp()
2016-10-25 22:50:11 +02:00
{
QDateTime ts;
2016-10-25 22:50:11 +02:00
while (_reader.readNextStartElement()) {
2016-10-27 22:33:35 +02:00
if (_reader.name() == "when")
ts = time();
2016-10-27 22:33:35 +02:00
else
2016-10-25 22:50:11 +02:00
_reader.skipCurrentElement();
}
return ts;
}
void KMLParser::lineString(TrackData &track)
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "coordinates") {
if (!lineCoordinates(track))
2016-10-29 12:22:28 +02:00
_reader.raiseError("Invalid coordinates");
} else
_reader.skipCurrentElement();
}
}
2016-10-27 22:33:35 +02:00
void KMLParser::point(Waypoint &waypoint)
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "coordinates") {
2016-10-27 22:33:35 +02:00
if (!pointCoordinates(waypoint))
2016-10-29 12:22:28 +02:00
_reader.raiseError("Invalid coordinates");
} else
_reader.skipCurrentElement();
}
2016-10-29 10:40:30 +02:00
if (waypoint.coordinates().isNull())
2016-10-29 12:22:28 +02:00
_reader.raiseError("Missing Point coordinates");
}
qreal KMLParser::value()
{
bool res;
qreal val = _reader.readElementText().toDouble(&res);
if (!res)
_reader.raiseError("Invalid value");
return val;
}
void KMLParser::heartRate(TrackData &track, int start)
{
int i = start;
const char error[] = "Heartrate data count mismatch";
while (_reader.readNextStartElement()) {
if (_reader.name() == "value") {
if (i < track.size())
track[i++].setCadence(value());
else {
_reader.raiseError(error);
return;
}
} else
_reader.skipCurrentElement();
}
if (i != track.size())
_reader.raiseError(error);
}
void KMLParser::cadence(TrackData &track, int start)
{
int i = start;
const char error[] = "Cadence data count mismatch";
while (_reader.readNextStartElement()) {
if (_reader.name() == "value") {
if (i < track.size())
track[i++].setCadence(value());
else {
_reader.raiseError(error);
return;
}
} else
_reader.skipCurrentElement();
}
if (i != track.size())
_reader.raiseError(error);
}
void KMLParser::speed(TrackData &track, int start)
{
int i = start;
const char error[] = "Speed data count mismatch";
while (_reader.readNextStartElement()) {
if (_reader.name() == "value") {
if (i < track.size())
track[i++].setSpeed(value());
else {
_reader.raiseError(error);
return;
}
} else
_reader.skipCurrentElement();
}
if (i != track.size())
_reader.raiseError(error);
}
void KMLParser::temperature(TrackData &track, int start)
{
int i = start;
const char error[] = "Temperature data count mismatch";
while (_reader.readNextStartElement()) {
if (_reader.name() == "value") {
if (i < track.size())
track[i++].setTemperature(value());
else {
_reader.raiseError(error);
return;
}
} else
_reader.skipCurrentElement();
}
if (i != track.size())
_reader.raiseError(error);
}
void KMLParser::schemaData(TrackData &track, int start)
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "SimpleArrayData") {
QXmlStreamAttributes attr = _reader.attributes();
QStringRef name = attr.value("name");
if (name == "Heartrate")
heartRate(track, start);
else if (name == "Cadence")
cadence(track, start);
else if (name == "Speed")
speed(track, start);
else if (name == "Temperature")
temperature(track, start);
else
_reader.skipCurrentElement();
} else
_reader.skipCurrentElement();
}
}
void KMLParser::extendedData(TrackData &track, int start)
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "SchemaData")
schemaData(track, start);
else
_reader.skipCurrentElement();
}
}
void KMLParser::track(TrackData &track)
2016-10-28 19:12:40 +02:00
{
const char error[] = "gx:coord/when element count mismatch";
int first = track.size();
int i = first;
2016-10-28 19:12:40 +02:00
while (_reader.readNextStartElement()) {
if (_reader.name() == "when") {
track.append(Trackpoint());
track.last().setTimestamp(time());
} else if (_reader.name() == "coord") {
if (i == track.size()) {
_reader.raiseError(error);
2016-10-28 19:12:40 +02:00
return;
} else if (!coord(track[i])) {
2016-10-29 12:22:28 +02:00
_reader.raiseError("Invalid coordinates");
2016-10-28 19:12:40 +02:00
return;
}
i++;
} else if (_reader.name() == "ExtendedData")
extendedData(track, first);
else
2016-10-28 19:12:40 +02:00
_reader.skipCurrentElement();
}
if (i != track.size())
_reader.raiseError(error);
2016-10-28 19:12:40 +02:00
}
void KMLParser::multiTrack(TrackData &t)
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "Track")
track(t);
else
_reader.skipCurrentElement();
}
}
2017-07-27 19:47:46 +02:00
void KMLParser::multiGeometry(QList<TrackData> &tracks,
QList<Waypoint> &waypoints, const QString &name, const QString &desc,
const QDateTime timestamp)
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "Point") {
2017-07-27 19:47:46 +02:00
waypoints.append(Waypoint());
Waypoint &w = waypoints.last();
w.setName(name);
w.setDescription(desc);
w.setTimestamp(timestamp);
point(w);
} else if (_reader.name() == "LineString") {
2017-07-27 19:47:46 +02:00
tracks.append(TrackData());
TrackData &t = tracks.last();
t.setName(name);
t.setDescription(desc);
lineString(t);
} else
_reader.skipCurrentElement();
}
}
2017-07-27 19:47:46 +02:00
void KMLParser::placemark(QList<TrackData> &tracks, QList<Waypoint> &waypoints)
{
QString name, desc;
QDateTime timestamp;
2016-10-25 22:50:11 +02:00
while (_reader.readNextStartElement()) {
2016-10-25 22:50:11 +02:00
if (_reader.name() == "name")
name = _reader.readElementText();
2016-10-25 22:50:11 +02:00
else if (_reader.name() == "description")
desc = _reader.readElementText();
2016-10-25 22:50:11 +02:00
else if (_reader.name() == "TimeStamp")
timestamp = timeStamp();
else if (_reader.name() == "MultiGeometry")
2017-07-27 19:47:46 +02:00
multiGeometry(tracks, waypoints, name, desc, timestamp);
2016-10-27 22:33:35 +02:00
else if (_reader.name() == "Point") {
2017-07-27 19:47:46 +02:00
waypoints.append(Waypoint());
Waypoint &w = waypoints.last();
w.setName(name);
w.setDescription(desc);
w.setTimestamp(timestamp);
point(w);
2017-07-30 12:13:45 +02:00
} else if (_reader.name() == "LineString"
|| _reader.name() == "LinearRing") {
2017-07-27 19:47:46 +02:00
tracks.append(TrackData());
TrackData &t = tracks.last();
t.setName(name);
t.setDescription(desc);
lineString(t);
2016-10-28 19:12:40 +02:00
} else if (_reader.name() == "Track") {
2017-07-27 19:47:46 +02:00
tracks.append(TrackData());
TrackData &t = tracks.last();
t.setName(name);
t.setDescription(desc);
track(t);
} else if (_reader.name() == "MultiTrack") {
tracks.append(TrackData());
TrackData &t = tracks.last();
t.setName(name);
t.setDescription(desc);
multiTrack(t);
2016-10-27 22:33:35 +02:00
} else
2016-10-25 22:50:11 +02:00
_reader.skipCurrentElement();
}
}
2017-07-27 19:47:46 +02:00
void KMLParser::folder(QList<TrackData> &tracks, QList<Waypoint> &waypoints)
2016-10-25 22:50:11 +02:00
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "Placemark")
2017-07-27 19:47:46 +02:00
placemark(tracks, waypoints);
2016-10-26 08:23:18 +02:00
else if (_reader.name() == "Folder")
2017-07-27 19:47:46 +02:00
folder(tracks, waypoints);
else
_reader.skipCurrentElement();
}
}
2017-07-27 19:47:46 +02:00
void KMLParser::document(QList<TrackData> &tracks, QList<Waypoint> &waypoints)
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "Placemark")
2017-07-27 19:47:46 +02:00
placemark(tracks, waypoints);
2016-10-25 22:50:11 +02:00
else if (_reader.name() == "Folder")
2017-07-27 19:47:46 +02:00
folder(tracks, waypoints);
else
_reader.skipCurrentElement();
}
}
2017-07-27 19:47:46 +02:00
void KMLParser::kml(QList<TrackData> &tracks, QList<Waypoint> &waypoints)
{
while (_reader.readNextStartElement()) {
if (_reader.name() == "Document")
2017-07-27 19:47:46 +02:00
document(tracks, waypoints);
else if (_reader.name() == "Placemark")
2017-07-27 19:47:46 +02:00
placemark(tracks, waypoints);
else if (_reader.name() == "Folder")
folder(tracks, waypoints);
else
_reader.skipCurrentElement();
}
}
2017-07-27 19:47:46 +02:00
bool KMLParser::parse(QFile *file, QList<TrackData> &tracks,
QList<RouteData> &routes, QList<Waypoint> &waypoints)
{
2017-07-27 19:47:46 +02:00
Q_UNUSED(routes);
_reader.clear();
_reader.setDevice(file);
if (_reader.readNextStartElement()) {
if (_reader.name() == "kml")
2017-07-27 19:47:46 +02:00
kml(tracks, waypoints);
else
2016-10-29 12:22:28 +02:00
_reader.raiseError("Not a KML file");
}
return !_reader.error();
}