KPublicTransport

journey.cpp
1/*
2 SPDX-FileCopyrightText: 2018 Volker Krause <vkrause@kde.org>
3
4 SPDX-License-Identifier: LGPL-2.0-or-later
5*/
6
7#include "journey.h"
8
9#include "identifier_p.h"
10#include "journeyutil_p.h"
11#include "json_p.h"
12#include "datatypes_p.h"
13#include "logging.h"
14#include "mergeutil_p.h"
15#include "notesutil_p.h"
16#include "platformutils_p.h"
17#include "rentalvehicle.h"
18#include "rentalvehicleutil_p.h"
19#include "stopover.h"
20
21#include <KLocalizedString>
22
23#include <QDebug>
24#include <QVariant>
25
26using namespace Qt::Literals::StringLiterals;
27using namespace KPublicTransport;
28
29namespace KPublicTransport {
30
31class JourneySectionPrivate : public QSharedData
32{
33public:
34 [[nodiscard]] bool isValidIndex(qsizetype idx) const;
35
36 JourneySection::Mode mode = JourneySection::Invalid;
37 Stopover departure;
38 Stopover arrival;
39 int distance = 0;
40 Disruption::Effect disruptionEffect = Disruption::NormalService;
41 QStringList notes;
42 std::vector<Stopover> intermediateStops;
43 int co2Emission = -1;
44 RentalVehicle rentalVehicle;
45 Path path;
46 IndividualTransport individualTransport;
47 IdentifierSet ids;
48};
49
50class JourneyPrivate : public QSharedData
51{
52public:
53 std::vector<JourneySection> sections;
54};
55
56}
57
58KPUBLICTRANSPORT_MAKE_GADGET(JourneySection)
59KPUBLICTRANSPORT_MAKE_PROPERTY(JourneySection, JourneySection::Mode, mode, setMode)
60KPUBLICTRANSPORT_MAKE_PROPERTY(JourneySection, QStringList, notes, setNotes)
61KPUBLICTRANSPORT_MAKE_PROPERTY(JourneySection, Stopover, arrival, setArrival)
62KPUBLICTRANSPORT_MAKE_PROPERTY(JourneySection, RentalVehicle, rentalVehicle, setRentalVehicle)
63KPUBLICTRANSPORT_MAKE_PROPERTY(JourneySection, Path, path, setPath)
64KPUBLICTRANSPORT_MAKE_PROPERTY(JourneySection, KPublicTransport::IndividualTransport, individualTransport, setIndividualTransport)
65
66bool JourneySectionPrivate::isValidIndex(qsizetype idx) const
67{
68 return idx >= 0 && idx <= (qsizetype)(intermediateStops.size()) + 1;
69}
70
72{
73 return d->departure.scheduledDepartureTime();
74}
75
76void JourneySection::setScheduledDepartureTime(const QDateTime &value)
77{
78 d.detach();
79 d->departure.setScheduledDepartureTime(value);
80}
81
83{
84 return d->departure.expectedDepartureTime();
85}
86
87void JourneySection::setExpectedDepartureTime(const QDateTime &value)
88{
89 d.detach();
90 d->departure.setExpectedDepartureTime(value);
91}
92
94{
95 return d->departure.hasExpectedDepartureTime();
96}
97
99{
100 return d->departure.departureDelay();
101}
102
104{
105 return d->arrival.scheduledArrivalTime();
106}
107
108void JourneySection::setScheduledArrivalTime(const QDateTime &value)
109{
110 d.detach();
111 d->arrival.setScheduledArrivalTime(value);
112}
113
115{
116 return d->arrival.expectedArrivalTime();
117}
118
119void JourneySection::setExpectedArrivalTime(const QDateTime &value)
120{
121 d.detach();
122 d->arrival.setExpectedArrivalTime(value);
123}
124
126{
127 return d->arrival.hasExpectedArrivalTime();
128}
129
131{
132 return d->arrival.arrivalDelay();
133}
134
135int JourneySection::duration() const
136{
137 return d->departure.scheduledDepartureTime().secsTo(d->arrival.scheduledArrivalTime());
138}
139
141{
142 return d->departure.stopPoint();
143}
144
145void JourneySection::setFrom(const Location &value)
146{
147 d.detach();
148 d->departure.setStopPoint(value);
149}
150
152{
153 return d->arrival.stopPoint();
154}
155
156void JourneySection::setTo(const Location &value)
157{
158 d.detach();
159 d->arrival.setStopPoint(value);
160}
161
162int JourneySection::distance() const
163{
164 if (d->mode == JourneySection::Waiting) {
165 return 0;
166 }
167
168 double dist = 0;
169 if (d->departure.stopPoint().hasCoordinate() && d->arrival.stopPoint().hasCoordinate()) {
170 auto startLat = d->departure.stopPoint().latitude();
171 auto startLon = d->departure.stopPoint().longitude();
172
173 for (const auto &stop : d->intermediateStops) {
174 if (!stop.stopPoint().hasCoordinate()) {
175 continue;
176 }
177 dist += Location::distance(startLat, startLon, stop.stopPoint().latitude(), stop.stopPoint().longitude());
178 startLat = stop.stopPoint().latitude();
179 startLon = stop.stopPoint().longitude();
180 }
181
182 dist += Location::distance(startLat, startLon, d->arrival.stopPoint().latitude(), d->arrival.stopPoint().longitude());
183 }
184 dist = std::max<double>(dist, d->path.distance());
185 return std::max((int)std::round(dist), d->distance);
186}
187
188void JourneySection::setDistance(int value)
189{
190 d.detach();
191 d->distance = value;
192}
193
195{
196 return d->departure.route();
197}
198
199void JourneySection::setRoute(const Route &value)
200{
201 d.detach();
202 d->departure.setRoute(value);
203}
204
206{
207 return d->departure.scheduledPlatform();
208}
209
210void JourneySection::setScheduledDeparturePlatform(const QString &platform)
211{
212 d.detach();
213 d->departure.setScheduledPlatform(platform);
214}
215
217{
218 return d->departure.expectedPlatform();
219}
220
221void JourneySection::setExpectedDeparturePlatform(const QString &platform)
222{
223 d.detach();
224 d->departure.setExpectedPlatform(platform);
225}
226
228{
229 return d->departure.hasExpectedPlatform();
230}
231
233{
234 return d->departure.platformChanged();
235}
236
238{
239 return d->arrival.scheduledPlatform();
240}
241
242void JourneySection::setScheduledArrivalPlatform(const QString &platform)
243{
244 d.detach();
245 d->arrival.setScheduledPlatform(platform);
246}
247
249{
250 return d->arrival.expectedPlatform();
251}
252
253void JourneySection::setExpectedArrivalPlatform(const QString &platform)
254{
255 d.detach();
256 d->arrival.setExpectedPlatform(platform);
257}
258
260{
261 return d->arrival.hasExpectedPlatform();
262}
263
265{
266 return d->arrival.platformChanged();
267}
268
270{
271 return std::max(d->disruptionEffect, std::max(d->departure.disruptionEffect(), d->arrival.disruptionEffect()));
272}
273
274void JourneySection::setDisruptionEffect(Disruption::Effect value)
275{
276 d.detach();
277 d->disruptionEffect = value;
278}
279
281{
282 const auto n = NotesUtil::normalizeNote(note);
283 const auto idx = NotesUtil::needsAdding(d->notes, n);
284 if (idx >= 0) {
285 d.detach();
286 NotesUtil::performAdd(d->notes, n, idx);
287 }
288}
289
290void JourneySection::addNotes(const QStringList &notes)
291{
292 for (const auto &n : notes) {
293 addNote(n);
294 }
295}
296
297const std::vector<Stopover>& JourneySection::intermediateStops() const
298{
299 return d->intermediateStops;
300}
301
303{
304 d.detach();
305 return std::move(d->intermediateStops);
306}
307
308void JourneySection::setIntermediateStops(std::vector<Stopover> &&stops)
309{
310 d.detach();
311 d->intermediateStops = std::move(stops);
312}
313
314QVariantList JourneySection::intermediateStopsVariant() const
315{
316 QVariantList l;
317 l.reserve(d->intermediateStops.size());
318 std::transform(d->intermediateStops.begin(), d->intermediateStops.end(), std::back_inserter(l), [](const auto &stop) { return QVariant::fromValue(stop); });
319 return l;
320}
321
323{
324 return d->departure;
325}
326
327void JourneySection::setDeparture(const Stopover &departure)
328{
329 d.detach();
330 auto dep = departure;
331 if (departure.route().line().mode() == Line::Unknown) {
332 dep.setRoute(route());
333 }
334 if (departure.loadInformation().empty()) {
335 dep.setLoadInformation(d->departure.takeLoadInformation());
336 }
337 d->departure = dep;
338}
339
340struct {
341 Line::Mode mode;
342 int gramPerKm;
343} static constexpr const emissionForModeMap[] = {
344 { Line::Air, 285 },
345 { Line::Boat, 245 },
346 { Line::Bus, 68 },
347 { Line::Coach, 68 },
348 { Line::Ferry, 245 },
349 // { Line::Funicular, -1 }, TODO
350 { Line::LocalTrain, 14 },
351 { Line::LongDistanceTrain, 14 },
352 { Line::Metro, 11 },
353 { Line::RailShuttle, 11 }, // assuming tram/rapid transit-like
354 { Line::RapidTransit, 11 },
355 { Line::Shuttle, 68 },
356 { Line::Taxi, 158 },
357 { Line::Train, 14 },
358 { Line::Tramway, 11 },
359 { Line::RideShare, 158 },
360 // { Line::AerialLift, -1 }, TODO
361};
362
363struct {
365 int gramPerKm;
366} static constexpr const emissionForIvModeMap[] = {
367 { IndividualTransport::Walk, 0 },
368 { IndividualTransport::Bike, 0 },
369 { IndividualTransport::Car, 158 }
370};
371
372struct {
374 int gramPerKm;
375} static constexpr const emissionForRvModeMap[] = {
377 // { RentalVehicle::Pedelec, -1 }, TODO
378 // { RentalVehicle::ElectricKickScooter, -1 }, TODO
379 // { RentalVehicle::ElectricMoped, -1 }, TODO
380 { RentalVehicle::Car, 158 },
381};
382
384{
385 if (d->co2Emission >= 0) {
386 return d->co2Emission;
387 }
388
389 switch (d->mode) {
390 case JourneySection::Invalid:
391 return -1;
392 case JourneySection::Walking:
393 case JourneySection::Transfer:
394 case JourneySection::Waiting:
395 return 0;
396 case JourneySection::PublicTransport:
397 {
398 const auto mode = route().line().mode();
399 for (const auto &map : emissionForModeMap) {
400 if (map.mode == mode) {
401 return (map.gramPerKm * distance()) / 1000;
402 }
403 }
404 qCDebug(Log) << "No CO2 emission estimate for mode" << mode;
405 return -1;
406 }
408 {
409 const auto mode = individualTransport().mode();
410 for (const auto &map :emissionForIvModeMap) {
411 if (map.mode == mode) {
412 return (map.gramPerKm *distance()) / 1000;
413 }
414 }
415 qCDebug(Log) << "No CO2 emission estimate for mode" << mode;
416 return -1;
417 }
419 {
420 const auto mode = rentalVehicle().type();
421 for (const auto &map :emissionForRvModeMap) {
422 if (map.mode == mode) {
423 return (map.gramPerKm *distance()) / 1000;
424 }
425 }
426 qCDebug(Log) << "No CO2 emission estimate for vehicle type" << mode;
427 return -1;
428 }
429 }
430
431 return -1;
432}
433
434void JourneySection::setCo2Emission(int value)
435{
436 d.detach();
437 d->co2Emission = value;
438}
439
440const std::vector<LoadInfo>& JourneySection::loadInformation() const
441{
442 return d->departure.loadInformation();
443}
444
445void JourneySection::setLoadInformation(std::vector<LoadInfo> &&loadInfo)
446{
447 d.detach();
448 d->departure.setLoadInformation(std::move(loadInfo));
449}
450
451QList<LoadInfo> JourneySection::loadInformationList() const
452{
454 l.reserve((qsizetype)d->departure.loadInformation().size());
455 std::ranges::copy(d->departure.loadInformation(), std::back_inserter(l));
456 return l;
457}
458
459void JourneySection::setLoadInformationList(const QList<LoadInfo> &loadInfo)
460{
461 d.detach();
462 std::vector<LoadInfo> l;
463 l.reserve(loadInfo.size());
464 std::ranges::copy(loadInfo, std::back_inserter(l));
465 d->departure.setLoadInformation(std::move(l));
466}
467
469{
470 return d->departure.vehicleLayout();
471}
472
473void JourneySection::setDepartureVehicleLayout(const Vehicle &value)
474{
475 d.detach();
476 d->departure.setVehicleLayout(value);
477}
478
480{
481 return d->departure.platformLayout();
482}
483
484void JourneySection::setDeparturePlatformLayout(const Platform &value)
485{
486 d.detach();
487 d->departure.setPlatformLayout(value);
488}
489
491{
492 return d->arrival.vehicleLayout();
493}
494
495void JourneySection::setArrivalVehicleLayout(const Vehicle &value)
496{
497 d.detach();
498 d->arrival.setVehicleLayout(value);
499}
500
502{
503 return d->arrival.platformLayout();
504}
505
506void JourneySection::setArrivalPlatformLayout(const Platform &value)
507{
508 d.detach();
509 d->arrival.setPlatformLayout(value);
510}
511
512const std::vector<KPublicTransport::Feature>& JourneySection::features() const
513{
514 return d->departure.vehicleLayout().features();
515}
516
517std::vector<KPublicTransport::Feature>&& JourneySection::takeFeatures()
518{
519 d.detach();
520 // TODO does this actually work?
521 return d->departure.vehicleLayout().takeFeatures();
522}
523
524void JourneySection::setFeatures(std::vector<KPublicTransport::Feature> &&features)
525{
526 d.detach();
527 auto l = d->departure.vehicleLayout();
528 l.setFeatures(std::move(features));
529 d->departure.setVehicleLayout(l);
530}
531
532QString JourneySection::iconName() const
533{
534 switch (d->mode) {
535 case JourneySection::Invalid:
536 return {};
537 case JourneySection::PublicTransport:
538 return d->departure.route().line().iconName();
540 return d->rentalVehicle.vehicleTypeIconName();
542 return d->individualTransport.modeIconName();
543 case JourneySection::Transfer:
544 case JourneySection::Walking:
545 case JourneySection::Waiting:
546 break;
547 }
548
549 return modeIconName(d->mode);
550}
551
553{
554 switch (mode) {
555 case JourneySection::Invalid:
556 return {};
557 case JourneySection::PublicTransport:
558 return Line::modeIconName(Line::Train);
559 case JourneySection::Transfer:
560 return u"qrc:///org.kde.kpublictransport/assets/images/journey-mode-transfer.svg"_s;
561 case JourneySection::Walking:
562 return IndividualTransport::modeIconName(IndividualTransport::Walk);
563 case JourneySection::Waiting:
564 return u"qrc:///org.kde.kpublictransport/assets/images/journey-mode-wait.svg"_s;
568 return IndividualTransport::modeIconName(IndividualTransport::Bike);
569 }
570
571 return u"question"_s;
572}
573
575{
576 switch (mode()) {
577 case JourneySection::Invalid:
578 break;
579 case JourneySection::PublicTransport:
580 return route().line().name();
581 case JourneySection::Walking:
582 return i18nc("mode of individual transport", "Walk");
583 case JourneySection::Waiting:
584 return i18n("Wait");
585 case JourneySection::Transfer:
586 break; // ?
588 return rentalVehicle().label();
590 return individualTransport().label();
591 }
592
593 return {};
594}
595
597{
598 return std::accumulate(d->departure.loadInformation().begin(), d->departure.loadInformation().end(), Load::Unknown, [](auto l, const auto &info) {
599 return std::max(l, info.load());
600 });
601}
602
604{
605 return d->ids.identifier(identifierType);
606}
607
608bool JourneySection::hasIdentifier(QAnyStringView identifierType) const
609{
610 return d->ids.hasIdentifier(identifierType);
611}
612
613void JourneySection::setIdentifier(const QString &identifierType, const QString &id)
614{
615 d.detach();
616 d->ids.setIdentifier(identifierType, id);
617}
618
620{
621 return !d->ids.isEmpty();
622}
623
625{
626 if (!d->isValidIndex(idx)) {
627 return {};
628 }
629
630 if (idx == 0) {
631 return departure();
632 }
633 if (idx <= (qsizetype)d->intermediateStops.size()) {
634 return d->intermediateStops[idx - 1];
635 }
636 return arrival();
637}
638
639void JourneySection::setStopovver(qsizetype idx, const Stopover &stop)
640{
641 if (!d->isValidIndex(idx)) {
642 return;
643 }
644
645 if (idx == 0) {
646 setDeparture(stop);
647 } else if (idx <= (qsizetype)d->intermediateStops.size()) {
648 d.detach();
649 d->intermediateStops[idx - 1] = stop;
650 } else {
651 setArrival(stop);
652 }
653}
654
656{
658 return 0;
659 }
660 if (const auto it = std::ranges::find_if(d->intermediateStops, [&stop](const auto &s) { return Stopover::isSame(s, stop); }); it != d->intermediateStops.end()) {
661 return std::distance(d->intermediateStops.begin(), it) + 1;
662 }
663 if (Stopover::isSame(arrival(), stop)) {
664 return (qsizetype)d->intermediateStops.size() + 1;
665 }
666 return -1;
667}
668
669JourneySection JourneySection::subsection(qsizetype begin, qsizetype end) const
670{
671 if (!d->isValidIndex(begin) || !d->isValidIndex(end) || end <= begin) {
672 return {};
673 }
674
675 auto partialTrip = *this;
676 if (begin > 0) {
677 partialTrip.setDeparture(stopover(begin));
678 }
679 if (end <= (qsizetype)d->intermediateStops.size()) {
680 partialTrip.setArrival(stopover(end));
681 }
682
683 const auto beginIt = d->intermediateStops.begin() + std::max<qsizetype>(0, begin);
684 const auto endIt = beginIt + std::max<qsizetype>(0, end - begin - 1);
685 partialTrip.setIntermediateStops({beginIt, endIt});
686
687 auto path = d->path;
688 auto pathSections = path.takeSections();
689
690 // find the closest points on the path to the departure/arrival locations
691 // we have to expect aribitrarly broken/imprecise data here...
692 qsizetype beginSection = -1, beginPoint = -1, endSection = -1, endPoint = -1;
693 double beginMinDist = std::numeric_limits<double>::max(), endMinDist = std::numeric_limits<double>::max();
694 for (std::size_t i = 0; i < pathSections.size(); ++i) {
695 const auto poly = pathSections[i].path();
696 for (qsizetype j = 0; j < poly.size(); ++j) {
697 const auto p = poly[j];
698 if (const auto d = Location::distance(p.y(), p.x(), partialTrip.from().latitude(), partialTrip.from().longitude()); d <beginMinDist) {
699 beginSection = (qsizetype)i;
700 beginPoint = j;
701 beginMinDist = d;
702 }
703 if (const auto d = Location::distance(p.y(), p.x(), partialTrip.to().latitude(), partialTrip.to().longitude()); d <endMinDist) {
704 endSection = (qsizetype)i;
705 endPoint = j;
706 endMinDist = d;
707 }
708 }
709 }
710
711 // if we found something, truncate path accordingly
712 if (std::tie(beginSection, beginPoint) < std::tie(endSection, endPoint)) {
713 // start cutting from the end, as otherwise the end indices become invalid!
714 pathSections.erase(pathSections.begin() + endSection + 1, pathSections.end());
715 pathSections.erase(pathSections.begin(), pathSections.begin() + beginSection);
716
717 auto poly = pathSections.back().path();
718 poly.erase(poly.begin() + endPoint + 1, poly.end());
719 pathSections.back().setPath(poly);
720
721 poly = pathSections.front().path();
722 poly.erase(poly.begin(), poly.begin() + beginPoint);
723 pathSections.front().setPath(poly);
724
725 path.setSections(std::move(pathSections));
726 partialTrip.setPath(path);
727 }
728
729 return partialTrip;
730}
731
733{
734 if (!d->departure.stopPoint().hasCoordinate() || mode() != JourneySection::PublicTransport) {
735 return;
736 }
737 auto r = route();
738 auto line = d->departure.route().line();
739 line.applyMetaData(d->departure.stopPoint(), download);
740 r.setLine(line);
741 d->departure.setRoute(r);
742 d->arrival.setRoute(r);
743
744 // propagate to intermediate stops
745 for (auto &stop : d->intermediateStops) {
746 stop.setRoute(r);
747 }
748}
749
751{
752 if (lhs.d->mode != rhs.d->mode) {
753 return false;
754 }
755
757 return false;
758 }
759
760 switch (lhs.d->ids.compare(rhs.d->ids)) {
761 case IdentifierSet::NotEqual: return false;
762 case IdentifierSet::Equal: // same trip id can still mean diffierent departure/arrival stops
763 case IdentifierSet::NoIntersection: break;
764 }
765
766 // we have N criteria to compare here, with 3 possible results:
767 // - equal
768 // - similar-ish, unknwon, or at least not conflicting
769 // - conflicting
770 // A single conflict results in a negative result, at least N - 1 equal comparisons lead to
771 // in a positive result.
772 enum { Equal = 1, Compatible = 0, Conflict = -1000 };
773 int result = 0;
774
775 const auto depTimeDist = MergeUtil::distance(lhs.scheduledDepartureTime(), rhs.scheduledDepartureTime());
776 result += depTimeDist < 60 ? Equal : depTimeDist <= 60 ? Compatible : Conflict;
777 const auto arrTimeDist = MergeUtil::distance(lhs.scheduledArrivalTime(), rhs.scheduledArrivalTime());
778 result += arrTimeDist < 60 ? Equal : depTimeDist <= 60 ? Compatible : Conflict;
779
780 const auto sameFrom = Location::isSame(lhs.from(), rhs.from());
781 const auto fromDist = Location::distance(lhs.from(), rhs.from());
782 result += sameFrom ? Equal : fromDist < 200 ? Compatible : Conflict;
783
784 const auto sameTo = Location::isSame(lhs.to(), rhs.to());
785 const auto toDist = Location::distance(lhs.to(), rhs.to());
786 result += sameTo ? Equal : toDist < 200 ? Compatible : Conflict;
787
788 const auto sameRoute = Route::isSame(lhs.route(), rhs.route());
789 const auto sameDir = Location::isSameName(lhs.route().direction(), rhs.route().direction());
790 const auto sameLine = Line::isSame(lhs.route().line(), rhs.route().line());
791 result += sameRoute ? Equal : (sameDir || sameLine) ? Compatible : Conflict;
792
794 result += lhs.scheduledDeparturePlatform() == rhs.scheduledDeparturePlatform() ? Equal : Conflict;
795 }
796
797 return result >= 4;
798}
799
801{
802 using namespace MergeUtil;
803 auto res = lhs;
804 res.d.detach();
805 res.d->ids.merge(rhs.d->ids);
806 res.d->departure = Stopover::merge(lhs.d->departure, rhs.d->departure);
807 res.d->arrival = Stopover::merge(lhs.d->arrival, rhs.d->arrival);
808
809 res.setDisruptionEffect(std::max(lhs.disruptionEffect(), rhs.disruptionEffect()));
810 res.setNotes(NotesUtil::mergeNotes(lhs.notes(), rhs.notes()));
811 res.setDistance(std::max(lhs.d->distance, rhs.d->distance));
812
813 if (lhs.intermediateStops().size() == rhs.intermediateStops().size()) {
814 auto stops = res.takeIntermediateStops();
815 for (uint i = 0; i < stops.size(); ++i) {
816 stops[i] = Stopover::merge(stops[i], rhs.intermediateStops()[i]);
817 stops[i].setRoute(res.route());
818 }
819 res.setIntermediateStops(std::move(stops));
820 } else if (lhs.intermediateStops().empty() && !rhs.intermediateStops().empty()) {
821 res.setIntermediateStops(std::vector<Stopover>(rhs.intermediateStops()));
822 } else if (!lhs.intermediateStops().empty() && rhs.intermediateStops().empty()) {
823 res.setIntermediateStops(std::vector<Stopover>(lhs.intermediateStops()));
824 }
825
826 res.d->co2Emission = std::max(lhs.d->co2Emission, rhs.d->co2Emission);
827 res.d->rentalVehicle = RentalVehicleUtil::merge(lhs.d->rentalVehicle, rhs.d->rentalVehicle);
828
829 res.d->path = lhs.d->path.sections().size() < rhs.d->path.sections().size() ? rhs.d->path : lhs.d->path;
830
831 return res;
832}
833
835{
836 if (section.mode() == Invalid) {
837 return {};
838 }
839
840 auto obj = Json::toJson(section);
841 if (!section.d->ids.isEmpty()) {
842 obj.insert("identifiers"_L1, section.d->ids.toJson());
843 }
844 obj.insert("departure"_L1, Stopover::toJson(section.d->departure));
845 obj.insert("arrival"_L1, Stopover::toJson(section.d->arrival));
846 if (section.mode() == PublicTransport) {
847 if (!section.intermediateStops().empty()) {
848 obj.insert("intermediateStops"_L1, Stopover::toJson(section.intermediateStops()));
849 }
850 }
851 if (section.d->distance <= 0) {
852 obj.remove("distance"_L1);
853 }
854 if (section.d->co2Emission < 0) {
855 obj.remove("co2Emission"_L1);
856 }
857 if (section.d->disruptionEffect == Disruption::NormalService) {
858 obj.remove("disruptionEffect"_L1);
859 }
860 if (section.rentalVehicle().type() != RentalVehicle::Unknown) {
861 obj.insert("rentalVehicle"_L1, RentalVehicle::toJson(section.rentalVehicle()));
862 }
863
864 if (!section.path().isEmpty()) {
865 obj.insert("path"_L1, Path::toJson(section.path()));
866 }
867
868 if (section.mode() == JourneySection::IndividualTransport) {
869 obj.insert("individualTransport"_L1, IndividualTransport::toJson(section.individualTransport()));
870 }
871
872 if (obj.size() <= 1) { // only the mode enums, ie. this is an empty object
873 return {};
874 }
875 return obj;
876}
877
878QJsonArray JourneySection::toJson(const std::vector<JourneySection> &sections)
879{
880 return Json::toJson(sections);
881}
882
884{
885 JourneySection section;
886 section.d->ids.fromJson(obj.value("identifiers"_L1).toObject());
887 section.d->departure = Stopover::fromJson(obj.value("departure"_L1).toObject());
888 section.d->arrival = Stopover::fromJson(obj.value("arrival"_L1).toObject());
889 // apply legacy properties after setting departure/arrival, otherwise we overwrite those above
890 Json::fromJson(&JourneySection::staticMetaObject, obj, &section);
891 section.setIntermediateStops(Stopover::fromJson(obj.value("intermediateStops"_L1).toArray()));
892 section.setRentalVehicle(RentalVehicle::fromJson(obj.value("rentalVehicle"_L1).toObject()));
893 section.setPath(Path::fromJson(obj.value("path"_L1).toObject()));
894 section.setIndividualTransport(IndividualTransport::fromJson(obj.value("individualTransport"_L1).toObject()));
895
896 // legacy format
897 if (auto it = obj.find("from"_L1); it != obj.end()) {
898 section.setFrom(Location::fromJson((*it).toObject()));
899 }
900 if (auto it = obj.find("to"_L1); it != obj.end()) {
901 section.setTo(Location::fromJson((*it).toObject()));
902 }
903 if (auto it = obj.find("route"_L1); it != obj.end()) {
904 section.setRoute(Route::fromJson((*it).toObject()));
905 }
906 if (auto it = obj.find("load"_L1); it != obj.end()) {
907 section.setLoadInformation(LoadInfo::fromJson((*it).toArray()));
908 }
909 if (auto it = obj.find("departureVehicleLayout"_L1); it != obj.end()) {
910 section.setDepartureVehicleLayout(Vehicle::fromJson((*it).toObject()));
911 }
912 if (auto it = obj.find("departurePlatformLayout"_L1); it != obj.end()) {
913 section.setDeparturePlatformLayout(Platform::fromJson((*it).toObject()));
914 }
915 if (auto it = obj.find("arrivalVehicleLayout"_L1); it != obj.end()) {
916 section.setArrivalVehicleLayout(Vehicle::fromJson((*it).toObject()));
917 }
918 if (auto it = obj.find("arrivalPlatformLayout"_L1); it != obj.end()) {
919 section.setArrivalPlatformLayout(Platform::fromJson((*it).toObject()));
920 }
921
922 section.applyMetaData(false);
923 return section;
924}
925
926std::vector<JourneySection> JourneySection::fromJson(const QJsonArray &array)
927{
928 return Json::fromJson<JourneySection>(array);
929}
930
931
932KPUBLICTRANSPORT_MAKE_GADGET(Journey)
933
934const std::vector<JourneySection>& Journey::sections() const
935{
936 return d->sections;
937}
938
939std::vector<JourneySection>&& Journey::takeSections()
940{
941 d.detach();
942 return std::move(d->sections);
943}
944
945void Journey::setSections(std::vector<JourneySection> &&sections)
946{
947 d.detach();
948 d->sections = std::move(sections);
949}
950
951QList<JourneySection> Journey::sectionsList() const
952{
954 l.reserve((qsizetype)d->sections.size());
955 std::copy(d->sections.begin(), d->sections.end(), std::back_inserter(l));
956 return l;
957}
958
959void Journey::setSectionsList(const QList<JourneySection> &sections)
960{
961 d.detach();
962 d->sections.clear();
963 d->sections.reserve(sections.size());
964 std::copy(sections.begin(), sections.end(), std::back_inserter(d->sections));
965}
966
967QDateTime Journey::scheduledDepartureTime() const
968{
969 if (!d->sections.empty()) {
970 return d->sections.front().scheduledDepartureTime();
971 }
972 return {};
973}
974
976{
977 return d->sections.empty() ? false : d->sections.front().hasExpectedDepartureTime();
978}
979
980QDateTime Journey::expectedDepartureTime() const
981{
982 return d->sections.empty() ? QDateTime() : d->sections.front().expectedDepartureTime();
983}
984
985int Journey::departureDelay() const
986{
987 return d->sections.empty() ? 0 : d->sections.front().departureDelay();
988}
989
990QDateTime Journey::scheduledArrivalTime() const
991{
992 if (!d->sections.empty()) {
993 return d->sections.back().scheduledArrivalTime();
994 }
995 return {};
996}
997
999{
1000 return d->sections.empty() ? false : d->sections.back().hasExpectedArrivalTime();
1001}
1002
1003QDateTime Journey::expectedArrivalTime() const
1004{
1005 return d->sections.empty() ? QDateTime() : d->sections.back().expectedArrivalTime();
1006}
1007
1008int Journey::arrivalDelay() const
1009{
1010 return d->sections.empty() ? 0 : d->sections.back().arrivalDelay();
1011}
1012
1013int Journey::duration() const
1014{
1016}
1017
1018int Journey::numberOfChanges() const
1019{
1020 return std::max(0, static_cast<int>(std::count_if(d->sections.begin(), d->sections.end(), [](const auto &section) { return section.mode() == JourneySection::PublicTransport; }) - 1));
1021}
1022
1024{
1025 Disruption::Effect effect = Disruption::NormalService;
1026 for (const auto &sec : d->sections) {
1027 effect = std::max(effect, sec.disruptionEffect());
1028 }
1029 return effect;
1030}
1031
1032int Journey::distance() const
1033{
1034 return std::accumulate(d->sections.begin(), d->sections.end(), 0, [](auto dist, const auto &jny) { return dist + jny.distance(); });
1035}
1036
1037int Journey::co2Emission() const
1038{
1039 return std::accumulate(d->sections.begin(), d->sections.end(), 0, [](auto co2, const auto &jny) { return co2 + std::max(0, jny.co2Emission()); });
1040}
1041
1043{
1044 return std::accumulate(d->sections.begin(), d->sections.end(), Load::Unknown, [](auto l, const auto &jny) {
1045 return std::max(l, jny.maximumOccupancy());
1046 });
1047}
1048
1049void Journey::applyMetaData(bool download)
1050{
1051 for (auto &sec : d->sections) {
1052 sec.applyMetaData(download);
1053 }
1054}
1055
1056static bool isTransportSection(JourneySection::Mode mode)
1057{
1058 return mode == JourneySection::PublicTransport
1061}
1062
1063bool Journey::isSame(const Journey &lhs, const Journey &rhs)
1064{
1065 auto lIt = lhs.sections().begin();
1066 auto rIt = rhs.sections().begin();
1067
1068 while (lIt != lhs.sections().end() || rIt != rhs.sections().end()) {
1069 // ignore non-transport sections
1070 if (lIt != lhs.sections().end() && !isTransportSection((*lIt).mode())) {
1071 ++lIt;
1072 continue;
1073 }
1074 if (rIt != rhs.sections().end() && !isTransportSection((*rIt).mode())) {
1075 ++rIt;
1076 continue;
1077 }
1078
1079 if (lIt == lhs.sections().end() || rIt == rhs.sections().end()) {
1080 return false;
1081 }
1082
1083 if (!JourneySection::isSame(*lIt, *rIt)) {
1084 return false;
1085 }
1086
1087 ++lIt;
1088 ++rIt;
1089 }
1090
1091 Q_ASSERT(lIt == lhs.sections().end() && rIt == rhs.sections().end());
1092 return true;
1093}
1094
1095Journey Journey::merge(const Journey &lhs, const Journey &rhs)
1096{
1097 std::vector<JourneySection> sections;
1098 sections.reserve(lhs.sections().size() + rhs.sections().size());
1099 std::copy(lhs.sections().begin(), lhs.sections().end(), std::back_inserter(sections));
1100 std::copy(rhs.sections().begin(), rhs.sections().end(), std::back_inserter(sections));
1101 std::sort(sections.begin(), sections.end(), [](const auto &lSec, const auto &rSec) {
1102 if (MergeUtil::distance(lSec.scheduledDepartureTime(), rSec.scheduledDepartureTime()) == 0) {
1103 return lSec.mode() < rSec.mode();
1104 }
1105 return MergeUtil::isBefore(lSec.scheduledDepartureTime(), rSec.scheduledDepartureTime());
1106 });
1107
1108 for (auto it = sections.begin(); it != sections.end(); ++it) {
1109 const auto nextIt = it + 1;
1110 if (nextIt == sections.end()) {
1111 break;
1112 }
1113
1114 if (JourneySection::isSame(*it, *nextIt) || ((*it).mode() == (*nextIt).mode() && (*it).mode() != JourneySection::PublicTransport)) {
1115 *it = JourneySection::merge(*it, *nextIt);
1116 sections.erase(nextIt);
1117 }
1118 }
1119
1120 Journey res;
1121 res.setSections(std::move(sections));
1122 return res;
1123}
1124
1126{
1127 QJsonObject obj;
1128 obj.insert(QLatin1String("sections"), JourneySection::toJson(journey.sections()));
1129 return obj;
1130}
1131
1132QJsonArray Journey::toJson(const std::vector<Journey> &journeys)
1133{
1134 return Json::toJson(journeys);
1135}
1136
1138{
1139 Journey j;
1140 j.setSections(JourneySection::fromJson(obj.value(QLatin1String("sections")).toArray()));
1141 return j;
1142}
1143
1144std::vector<Journey> Journey::fromJson(const QJsonArray &array)
1145{
1146 return Json::fromJson<Journey>(array);
1147}
1148
1149#include "moc_journey.cpp"
Individual transport mode details for a journey section, and for specifying journey requests.
QString label
Label shortly describing this transport for display.
Mode
Mode of (individual) transportation.
static QJsonObject toJson(const IndividualTransport &it)
Serializes one object to JSON.
static IndividualTransport fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
QString modeIconName
Name of an icon to represent this transport mode.
A segment of a journey plan.
Definition journey.h:39
KPublicTransport::Vehicle departureVehicleLayout
Vehicle coach layout information at departure.
Definition journey.h:157
QString scheduledArrivalPlatform
Planned arrival platform.
Definition journey.h:106
KPublicTransport::Path path
Movement path for this journey section.
Definition journey.h:154
void applyMetaData(bool download)
Augment line meta data.
Definition journey.cpp:732
bool arrivalPlatformChanged
true if we have real-time platform information and the platform changed.
Definition journey.h:112
static JourneySection merge(const JourneySection &lhs, const JourneySection &rhs)
Merge two instances.
Definition journey.cpp:800
Stopover stopover(qsizetype idx) const
Retrieve stopover at index idx.
Definition journey.cpp:624
qsizetype indexOfStopover(const Stopover &stop) const
Returns the index of stop in this journey section.
Definition journey.cpp:655
KPublicTransport::Load::Category maximumOccupancy
Maximum occpancy over all classes.
Definition journey.h:186
KPublicTransport::Location from
Departure location of this segment.
Definition journey.h:90
static bool isSame(const JourneySection &lhs, const JourneySection &rhs)
Checks if two instances refer to the same journey section (which does not necessarily mean they are e...
Definition journey.cpp:750
void setLoadInformation(std::vector< LoadInfo > &&loadInfo)
Set the vehicle load information for this journey section.
Definition journey.cpp:445
QDateTime expectedArrivalTime
Actual arrival time, if available.
Definition journey.h:78
KPublicTransport::Route route
Route to take on this segment.
Definition journey.h:94
QString iconName
The best available icon to represent this journey section.
Definition journey.h:180
QList< KPublicTransport::LoadInfo > loadInformation
Vehicle load information for this journey section.
Definition journey.h:145
int departureDelay
Difference to schedule in minutes.
Definition journey.h:71
JourneySection subsection(qsizetype begin, qsizetype end) const
Returns the sub-journey starting from index begin until end (inclusive).
Definition journey.cpp:669
void setIntermediateStops(std::vector< Stopover > &&stops)
Set the intermediate stops.
Definition journey.cpp:308
KPublicTransport::Platform arrivalPlatformLayout
Platform layout information at arrival.
Definition journey.h:166
QString label
Label shortly describing this transport for display.
Definition journey.h:183
static Q_INVOKABLE QString modeIconName(KPublicTransport::JourneySection::Mode mode)
Icon representing the journey section mode mode.
Definition journey.cpp:552
QString scheduledDeparturePlatform
Planned departure platform.
Definition journey.h:97
int co2Emission
CO₂ emission during this journey section, in gram.
Definition journey.h:140
KPublicTransport::RentalVehicle rentalVehicle
Information about a rental vehicle, for sections using one.
Definition journey.h:148
bool hasIdentifiers() const
Returns true if there is any identifier set at all.
Definition journey.cpp:619
QString identifier(QAnyStringView identifierType) const
Backend-specific journey section identifiers.
Definition journey.cpp:603
QStringList notes
General human-readable notes on this service, e.g.
Definition journey.h:117
static QJsonObject toJson(const JourneySection &section)
Serializes one journey section to JSON.
Definition journey.cpp:834
KPublicTransport::Stopover departure
The departure stopover of this journey section.
Definition journey.h:126
QDateTime scheduledDepartureTime
Planned departure time.
Definition journey.h:63
bool hasExpectedArrivalPlatform
true if real-time platform information are available.
Definition journey.h:110
KPublicTransport::Vehicle arrivalVehicleLayout
Vehicle coach layout information at arrival.
Definition journey.h:164
@ RentedVehicle
free floating or dock-based rental bike service, electric scooters, car sharing services,...
Definition journey.h:52
@ IndividualTransport
using your own vehicle (bike, car, etc).
Definition journey.h:53
KPublicTransport::Location to
Arrival location of this segment.
Definition journey.h:92
QVariantList intermediateStops
Intermediate stops for consumption by QML.
Definition journey.h:120
QString expectedDeparturePlatform
Actual departure platform, in case real-time information are available.
Definition journey.h:99
QString expectedArrivalPlatform
Actual arrival platform, in case real-time information are available.
Definition journey.h:108
void setStopovver(qsizetype idx, const Stopover &stop)
Set the stopover at index idx.
Definition journey.cpp:639
int arrivalDelay
Difference to schedule in minutes.
Definition journey.h:82
std::vector< Stopover > && takeIntermediateStops()
Moves the intermediate stops out of this object.
Definition journey.cpp:302
bool hasExpectedDeparturePlatform
true if real-time platform information are available.
Definition journey.h:101
KPublicTransport::Platform departurePlatformLayout
Platform layout information at departure.
Definition journey.h:159
bool hasExpectedDepartureTime
true if this has real-time data.
Definition journey.h:69
bool hasExpectedArrivalTime
true if this has real-time data.
Definition journey.h:80
static JourneySection fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
Definition journey.cpp:883
std::vector< KPublicTransport::Feature > features
Features of the vehicle used on this section.
Definition journey.h:175
void addNote(const QString &note)
Adds a note.
Definition journey.cpp:280
int duration
Duration of the section in seconds.
Definition journey.h:85
Mode mode
Mode of transport for this section.
Definition journey.h:60
QDateTime scheduledArrivalTime
Planned arrival time.
Definition journey.h:74
KPublicTransport::IndividualTransport individualTransport
Individual transport details for sections using your own vehicle.
Definition journey.h:169
bool departurePlatformChanged
true if we have real-time platform information and the platform changed.
Definition journey.h:103
QDateTime expectedDepartureTime
Actual departure time, if available.
Definition journey.h:67
KPublicTransport::Stopover arrival
The arrival stopover of this jouney section.
Definition journey.h:131
KPublicTransport::Disruption::Effect disruptionEffect
Disruption effect on this section, if any.
Definition journey.h:115
int distance
Distance of the section in meter.
Definition journey.h:87
A journey plan.
Definition journey.h:317
void applyMetaData(bool download)
Augment line meta data.
Definition journey.cpp:1049
KPublicTransport::Disruption::Effect disruptionEffect
Worst disruption effect of any of the journey sections.
Definition journey.h:348
static Journey fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
Definition journey.cpp:1137
QDateTime expectedDepartureTime
Actual departure time, if available.
Definition journey.h:328
QDateTime scheduledDepartureTime
Departure time of the journey, according to schedule.
Definition journey.h:322
static bool isSame(const Journey &lhs, const Journey &rhs)
Checks if two instances refer to the same journey (which does not necessarily mean they are exactly e...
Definition journey.cpp:1063
QDateTime expectedArrivalTime
Actual arrival time, if available.
Definition journey.h:339
int departureDelay
Difference to schedule in minutes.
Definition journey.h:330
int numberOfChanges
Number of changes on this journey.
Definition journey.h:346
void setSections(std::vector< JourneySection > &&sections)
Sets the journey sections.
Definition journey.cpp:945
KPublicTransport::Load::Category maximumOccupancy
Maximum occpancy in all journey sections, over all classes.
Definition journey.h:360
QList< KPublicTransport::JourneySection > sections
Journey sections for consumption by QML.
Definition journey.h:320
static QJsonObject toJson(const Journey &journey)
Serializes one journey object to JSON.
Definition journey.cpp:1125
bool hasExpectedDepartureTime
true if this has real-time data.
Definition journey.h:324
int duration
Duration of the entire journey in seconds.
Definition journey.h:344
QDateTime scheduledArrivalTime
Arrival time of the journey, according to schedule.
Definition journey.h:333
int distance
Total travelled distance of the entire journey in meter.
Definition journey.h:353
int arrivalDelay
Difference to schedule in minutes.
Definition journey.h:341
int co2Emission
Total CO2 emissions for the entire journey in gram.
Definition journey.h:357
std::vector< JourneySection > && takeSections()
Moves the journey sections out of this object.
Definition journey.cpp:939
static Journey merge(const Journey &lhs, const Journey &rhs)
Merge two instances.
Definition journey.cpp:1095
bool hasExpectedArrivalTime
true if this has real-time data.
Definition journey.h:335
KPublicTransport::Line::Mode mode
Type of transport.
Definition line.h:60
QString name
Name of the line.
Definition line.h:50
QString modeIconName
Generic icon for the line mode.
Definition line.h:93
Mode
Mode of transportation.
Definition line.h:27
@ RideShare
peer-to-peer ride sharing/car pooling
Definition line.h:44
@ RailShuttle
rail shuttle service within a complex, as e.g. found at or around airports
Definition line.h:38
@ Shuttle
shuttle bus/coach services, e.g. to/from an airport
Definition line.h:40
static bool isSame(const Line &lhs, const Line &rhs)
Checks if to instances refer to the same line (which does not necessarily mean they are exactly equal...
Definition line.cpp:167
static LoadInfo fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
Definition load.cpp:36
static Location fromJson(const QJsonObject &obj)
Deserialize a Location object from JSON.
Definition location.cpp:539
static bool isSameName(const QString &lhs, const QString &rhs)
Checks if two location names refer to the same location.
Definition location.cpp:360
static double distance(double lat1, double lon1, double lat2, double lon2)
Compute the distance between two geo coordinates, in meters.
Definition location.cpp:458
static bool isSame(const Location &lhs, const Location &rhs)
Checks if to instances refer to the same location (which does not necessarily mean they are exactly e...
Definition location.cpp:312
A path followed by any kind of location change.
Definition path.h:113
bool isEmpty() const
Returns true if this is an empty/not-set path.
Definition path.cpp:143
static Path fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
Definition path.cpp:187
static QJsonObject toJson(const Path &path)
Serializes one path object to JSON.
Definition path.cpp:180
std::vector< KPublicTransport::PathSection > sections
Access to path sections for QML.
Definition path.h:117
Information about the layout of a station platform.
Definition platform.h:45
static Platform fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
Definition platform.cpp:113
An individual rental vehicle used on a JourneySection, ie.
QString label
Label shortly describing this transport for display.
@ Car
electrical- or combustion-powered car
@ ElectricKickScooter
"e scooter", electrically assisted kick scooters, not to be confused with motorcycle-like scooters
@ Bicycle
human-powered bicylce
static RentalVehicle fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
static QJsonObject toJson(const RentalVehicle &vehicle)
Serializes one object to JSON.
VehicleType type
Vehicle type.
QString vehicleTypeIconName
Icon representing the vehicle type.
A route of a public transport line.
Definition line.h:148
QString direction
Direction of the route.
Definition line.h:157
static bool isSame(const Route &lhs, const Route &rhs)
Checks if to instances refer to the same route (which does not necessarily mean they are exactly equa...
Definition line.cpp:266
KPublicTransport::Line line
Line this route belongs to.
Definition line.h:151
static Route fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
Definition line.cpp:299
Information about an arrival and/or departure of a vehicle at a stop area.
Definition stopover.h:26
static Stopover merge(const Stopover &lhs, const Stopover &rhs)
Merge two departure instances.
Definition stopover.cpp:212
static QJsonObject toJson(const Stopover &stopover)
Serializes one object to JSON.
Definition stopover.cpp:239
static bool isSame(const Stopover &lhs, const Stopover &rhs)
Checks if to instances refer to the same departure (which does not necessarily mean they are exactly ...
Definition stopover.cpp:182
static Stopover fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
Definition stopover.cpp:271
Information about the vehicle used on a journey.
Definition vehicle.h:159
static Vehicle fromJson(const QJsonObject &obj)
Deserialize an object from JSON.
Definition vehicle.cpp:281
void stop(Ekos::AlignState mode)
QString i18nc(const char *context, const char *text, const TYPE &arg...)
QString i18n(const char *text, const TYPE &arg...)
Effect
Disruption effects, numerical sorted so that higher values imply more severe disruptions.
Definition disruption.h:25
Category
Vehicle load categories.
Definition load.h:20
@ Unknown
no load information are available
Definition load.h:21
Query operations and data types for accessing realtime public transport information from online servi...
QAction * back(const QObject *recvr, const char *slot, QObject *parent)
qint64 secsTo(const QDateTime &other) const const
iterator end()
iterator find(QLatin1StringView key)
iterator insert(QLatin1StringView key, const QJsonValue &value)
QJsonValue value(QLatin1StringView key) const const
QJsonArray toArray() const const
QJsonObject toObject() const const
iterator begin()
void clear()
iterator end()
iterator erase(const_iterator begin, const_iterator end)
void reserve(qsizetype size)
qsizetype size() const const
bool isEmpty() const const
QFuture< void > map(Iterator begin, Iterator end, MapFunctor &&function)
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 11:54:54 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.