Map Matching
track.cpp
Go to the documentation of this file.
1 #include "track.h"
2 #include <QLocale>
3 #include <limits>
4 #include <sstream>
5 
6 using namespace std;
7 
8 #define DEBUG_READCSV false
9 
11  : m_xMin(std::numeric_limits<double>::max())
12  , m_xMax(0.0)
13  , m_yMin(std::numeric_limits<double>::max())
14  , m_yMax(0.0)
15 {
16 }
17 #ifdef QT_DEBUG
18 Track::Track(const Track& oldTrack)
19  : m_xMin(oldTrack.m_xMin)
20  , m_xMax(oldTrack.m_xMax)
21  , m_yMin(oldTrack.m_yMin)
22  , m_yMax(oldTrack.m_yMax)
23 {
24 }
25 #endif
26 
28 {
29  for (uint i = 0; i < m_points.size(); ++i) {
30  delete m_points[i];
31  }
32  m_points.clear();
33 }
34 
35 // WARNING : This function doesn't check the correct extension (trust the user who puts a csv-format file)
36 void Track::readFromCSV(QString filename)
37 {
38  emit signalMessage("Loading track ...");
39  string value; // Save the value of the line
40  QString stringConverted;
41 
42  m_trackFullName = filename.toStdString();
43 
44  if (DEBUG_READCSV)
45  cout << "Le fichier " << filename.toStdString() << " va être lu." << endl;
46  ifstream file(filename.toStdString().c_str()); // c_str() :http://stackoverflow.com/questions/32332/why-dont-the-stdfstream-classes-take-a-stdstring
47  vector<int> correspondance(7); // Correpondance between the header and the parser :
48  // correspondance :
49  // 0 : # x
50  // 1 : # y
51  // 2 : # latitude column
52  // 3 : # longitude column
53  // 4 : # altitude column
54  // 5 : # date column
55  // 6 : # time column
56  for (uint i = 0; i < correspondance.size(); ++i) {
57  correspondance[i] = -1;
58  }
59  // Read header
60  if (file.good()) {
61  // Read the line
62  getline(file, value);
63 
64  // Convert string to Qstring (easiest
65  stringConverted = QString::fromStdString(value);
66 
67  // Split the line with separator
68  QStringList text = stringConverted.split(",");
69  //vector<string> text = split(value,',');
70 
71  // Parse header
72 
73  for (int i = 0; i < text.size(); ++i) {
74  if (DEBUG_READCSV)
75  cout << "Colonne " << text[i].toStdString() << endl;
76  if (text[i].contains(QString::fromStdString("X"), Qt::CaseInsensitive) && (!(text[i].contains(QString::fromStdString("max"), Qt::CaseInsensitive)))) {
77  correspondance[0] = i;
78  if (DEBUG_READCSV)
79  cout << "X DETECTED at colonne : " << i << endl;
80  } else if ((text[i].contains(QString::fromStdString("Y"), Qt::CaseInsensitive)) && (!(text[i].contains(QString::fromStdString("Type"), Qt::CaseInsensitive)))) {
81  correspondance[1] = i;
82  if (DEBUG_READCSV)
83  cout << "Y DETECTED at colonne : " << i << endl;
84  } else if (text[i].contains(QString::fromStdString("Lati"), Qt::CaseInsensitive)) {
85  correspondance[2] = i;
86  if (DEBUG_READCSV)
87  cout << "LATITUDE DETECTED at colonne : " << i << endl;
88  } else if (text[i].contains(QString::fromStdString("Longi"), Qt::CaseInsensitive)) {
89  correspondance[3] = i;
90  if (DEBUG_READCSV)
91  cout << "LONGITUDE DETECTED at colonne : " << i << endl;
92  } else if (text[i].contains(QString::fromStdString("Alti"), Qt::CaseInsensitive)) {
93  correspondance[4] = i;
94  if (DEBUG_READCSV)
95  cout << "ALTITUDE DETECTED at colonne : " << i << endl;
96  } else if (text[i].contains(QString::fromStdString("Date"), Qt::CaseInsensitive)) {
97  correspondance[5] = i;
98  if (DEBUG_READCSV)
99  cout << "DATE DETECTED at colonne : " << i << endl;
100  } else if ((text[i].contains(QString::fromStdString("Time"), Qt::CaseInsensitive)) || (text[i].contains(QString::fromStdString("Heure"), Qt::CaseInsensitive))) {
101  correspondance[6] = i;
102  if (DEBUG_READCSV)
103  cout << "Time DETECTED at colonne : " << i << endl;
104  } else {
105  if (DEBUG_READCSV)
106  cout << "Colonne " << text[i].toStdString() << " non reconnue" << endl;
107  }
108  }
109  }
110 
111  // Display the correspondance table
112  if (DEBUG_READCSV)
113  cout << "BEGIN CORRESPONDANCE" << endl;
114  for (uint i = 0; i < correspondance.size(); ++i) {
115  if (DEBUG_READCSV)
116  cout << correspondance[i] << endl;
117  }
118  if (DEBUG_READCSV)
119  cout << "END CORRESPONDANCE" << endl;
120 
121  // Parse the file
122  double x(0), y(0);
123  float latitude(0), longitude(0), altitude(0);
124  QDateTime ts(QDateTime::currentDateTime());
125  while (file.good()) {
126  // Reinitialize variables (default values)
127  x = 0;
128  y = 0;
129  latitude = 0;
130  longitude = 0;
131  altitude = 0;
132  ts = QDateTime::currentDateTime();
133  vector<QString> specificDate(2);
134 
135  getline(file, value);
136  stringConverted = QString::fromStdString(value);
137 
138  if (stringConverted.length() != 0) // Elimine les lignes vides
139  {
140  if (DEBUG_READCSV)
141  cout << "Read new point" << endl;
142  QStringList text = stringConverted.split(",");
143  for (int i = 0; i < text.size(); ++i) {
144  if (i == correspondance[0]) {
145  // Traitement X
146  // Read x from file
147  x = text[i].toDouble();
148 
149  if (DEBUG_READCSV)
150  cout << " X : " << x << " ";
151  } else if (i == correspondance[1]) {
152  // Traitement Y
153  // Read y from file
154  y = text[i].toDouble();
155 
156  if (DEBUG_READCSV)
157  cout << " Y : " << y << " ";
158  } else if (i == correspondance[2]) {
159  // Traitement Latitude
160  // Read latitude from file
161  latitude = text[i].toFloat();
162 
163  if (DEBUG_READCSV)
164  cout << " Latitude : " << latitude << " ";
165  } else if (i == correspondance[3]) {
166  // Traitement Longitude
167  // Read longitude from file
168  longitude = text[i].toFloat();
169 
170  if (DEBUG_READCSV)
171  cout << " Longitude : " << longitude << " ";
172  } else if (i == correspondance[4]) {
173  // Traitement Altitude
174  // Read altitude from file
175  altitude = text[i].toFloat();
176 
177  if (DEBUG_READCSV)
178  cout << " Altitude : " << altitude << " ";
179  } else if (i == correspondance[5]) {
180  // Traitement Date
181  // Read date from file
182  specificDate[0] = text[i];
183  if (DEBUG_READCSV)
184  cout << " Date : " << specificDate[0].toStdString();
185 
186  } else if (i == correspondance[6]) {
187  // Traitement Time
188  // Read time from file
189  specificDate[1] = text[i];
190  if (DEBUG_READCSV)
191  cout << " Time : " << specificDate[1].toStdString();
192  }
193  }
194  if ((specificDate[0].size() == 19) && (specificDate[1].size() == 0))
195  ts = QDateTime::fromString(specificDate[0], "yyyy-MM-dd hh:mm:ss");
196  else if ((specificDate[0].size() == 11) && (specificDate[1].size() == 0))
197  ts = QDateTime::fromString(specificDate[0], "yyyy-MM-dd");
198  else if ((specificDate[0].size() == 0) && (specificDate[1].size() == 8))
199  ts = QDateTime::fromString(specificDate[1], "hh:mm:ss");
200  else if ((specificDate[0].size() == 10) && (specificDate[1].size() == 8)) {
201  ts = QDateTime(QDate::fromString(specificDate[0], "yyyy/MM/dd"), QTime::fromString(specificDate[1], "hh:mm:ss"));
202  } else if ((specificDate[0].size() == 11) && (specificDate[1].size() == 8)) {
203  // Attention : si les mois sont enregistres avec "Jan",
204  // il faut passer en langue anglaise/americaine
205  // Sinon timeStamp est vide... et c'est embetant
206  QLocale locale(QLocale::English, QLocale::UnitedStates);
207  ts = locale.toDateTime((specificDate[0] + specificDate[1]), "dd'-'MMM'-'yyyyhh':'mm':'ss");
208  }
209 
210  if (DEBUG_READCSV)
211  cout << "timestamp : " << ts.toString("yyyy-MM-dd hh:mm:ss").toStdString();
212  if (DEBUG_READCSV)
213  cout << endl;
214  addPoint(x, y, altitude, ts.toTime_t());
215  } else {
216  if (DEBUG_READCSV)
217  cout << "Ligne ignorée" << endl;
218  continue;
219  }
220  }
222  if (DEBUG_READCSV)
223  cout << "Le fichier de traces GPS " << filename.toStdString() << " a été lu." << endl;
224 }
225 
226 vector<PointGPS*> Track::getPoints()
227 {
228  return m_points;
229 }
230 
231 void Track::temporalFilter(uint interval)
232 {
233  int initialNumberOfPoints(m_points.size());
234  emit signalMessage(QString("---\nTemporal filter initilizer ... ") + QString::number(initialNumberOfPoints) + QString(" initial points") + QString("\nTemporal filter will delete each successive point withing interval of ") + QString::number(interval) + QString(" seconds"));
235  PointGPS* pointPrecedent;
236 
237  bool firstElement(true);
238  for (vector<PointGPS*>::iterator b = m_points.begin(); b != m_points.end();) {
239  if (firstElement) {
240  pointPrecedent = *b;
241  firstElement = false;
242  ++b;
243  } else {
244  if ((*b)->timeStamp() - pointPrecedent->timeStamp() <= interval) {
245  b = m_points.erase(b); // reseat iterator to a valid value post-erase
246  } else {
247  pointPrecedent = *b;
248  ++b;
249  }
250  }
251  }
252 
253  int endNumberOfPoints(m_points.size());
254  double reduction(((double)(initialNumberOfPoints - endNumberOfPoints) * 100) / initialNumberOfPoints);
255  emit signalMessage(QString("Temporal filter ended ... ") + QString::number(endNumberOfPoints) + QString(" points left ") + QString(" ( ") + QString::number(reduction, 'g', 4) + QString(" % of reduction)"));
256 }
257 
258 void Track::spaceFilter(double interval)
259 {
260  int initialNumberOfPoints(m_points.size());
261  emit signalMessage(QString("---\nSpatial filter initilizer ... ") + QString::number(initialNumberOfPoints) + QString(" initial points") + QString("\nSpatial filter will delete each successive point withing interval of ") + QString::number(interval) + QString(" meters"));
262 
263  for (uint i = 0; i < m_points.size(); i++) { // on parcours la liste des points
264  // tant qu on ne se trouve pas sur le dernier point
265  //&& (security segmentation)
266  // tant que l interval est inferieure a la distance au point suivant
267  while (i != (m_points.size() - 1) && (interval > m_points[i]->distanceToPoint(*m_points[i + 1]))) {
268  this->delPointGPS(i + 1); // on supprime le point suivant
269  }
270  }
271  int endNumberOfPoints(m_points.size());
272  double reduction(((double)(initialNumberOfPoints - endNumberOfPoints) * 100) / initialNumberOfPoints);
273  emit signalMessage(QString("Spatial filter ended ... ") + QString::number(endNumberOfPoints) + QString(" points left ") + QString(" ( ") + QString::number(reduction, 'g', 4) + QString(" % of reduction)"));
274 }
275 
276 void Track::delPointGPS(int occurrence)
277 {
278  m_points.erase(m_points.begin() + occurrence);
279 }
280 
281 void Track::addPoint(double x, double y, float altitude, unsigned int timeStamp)
282 {
283  updateBox(x, y);
284  m_points.push_back(new PointGPS(x, y, altitude, timeStamp));
285 }
286 
287 void Track::updateBox(double x, double y)
288 {
289  m_xMin = std::min(m_xMin, x);
290  m_xMax = std::max(m_xMax, x);
291  m_yMin = std::min(m_yMin, y);
292  m_yMax = std::max(m_yMax, y);
293 }
294 
296 {
301 }
302 
304 {
305  cout << "Track " << m_trackFullName << " contains: \n\t" << m_points.size() << " points"
306  << "\n\tin box:";
307  cout << "\n\t\t min (x,y) (" << m_xMin << ", " << m_yMin << ")";
308  cout << "\n\t\t max (x,y) (" << m_xMax << ", " << m_yMax << ")" << endl;
309 }
310 
311 string Track::infos()
312 {
313  std::stringstream ss;
314  ss << "Track " << m_trackFullName << " contains: \n\t" << m_points.size() << " points"
315  << "\n\tin box:";
316  ss << "\n\t\t min (x,y) (" << m_xMin << ", " << m_yMin << ")";
317  ss << "\n\t\t max (x,y) (" << m_xMax << ", " << m_yMax << ")\n";
318  return ss.str();
319 }
#define DEBUG_READCSV
Definition: track.cpp:8
double m_xMin
Definition: track.h:107
void temporalFilter(uint interval)
This is a temporal filter, which deletes points depending on a time value.
Definition: track.cpp:231
STL namespace.
void readFromCSV(QString filename)
Reads a csv file and inserts each point in m_points vector.
Definition: track.cpp:36
std::vector< PointGPS * > getPoints()
getPoints Get the m_points vector
Definition: track.cpp:226
void updateBox(double x, double y)
Update the enclosing box of the track.
Definition: track.cpp:287
The Track class.
Definition: track.h:26
#define DISTANCE_THRESHOLD
Definition: track.h:21
double m_yMin
Definition: track.h:109
void outputInfos()
Print some informations to user in console.
Definition: track.cpp:303
Track()
Track&#39;s Constructor.
Definition: track.cpp:10
void spaceFilter(double interval)
This is a space filter, which deletes points depending on a distance interval.
Definition: track.cpp:258
virtual ~Track()
Definition: track.cpp:27
void addPoint(double x, double y, float altitude, unsigned int timeStamp)
Creates a new point and inserts it in m_points.
Definition: track.cpp:281
unsigned int timeStamp() const
Definition: pointGPS.h:39
void applyThresholdToBox()
Apply a threshold to bounding box once the data file has been read.
Definition: track.cpp:295
double m_yMax
Definition: track.h:110
void delPointGPS(int occurrence)
Deletes a occurence.
Definition: track.cpp:276
void signalMessage(QString)
std::string infos()
Return a string containing information about the content of this object.
Definition: track.cpp:311
double m_xMax
Definition: track.h:108
The class Track.
std::vector< PointGPS * > m_points
m_points Vector where points of the Track are saved
Definition: track.h:120
std::string m_trackFullName
Definition: track.h:122