Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
eclipse
GitHub Repository: eclipse/sumo
Path: blob/main/src/guisim/GUIBaseVehicle.cpp
169666 views
1
/****************************************************************************/
2
// Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
3
// Copyright (C) 2001-2025 German Aerospace Center (DLR) and others.
4
// This program and the accompanying materials are made available under the
5
// terms of the Eclipse Public License 2.0 which is available at
6
// https://www.eclipse.org/legal/epl-2.0/
7
// This Source Code may also be made available under the following Secondary
8
// Licenses when the conditions for such availability set forth in the Eclipse
9
// Public License 2.0 are satisfied: GNU General Public License, version 2
10
// or later which is available at
11
// https://www.gnu.org/licenses/old-licenses/gpl-2.0-standalone.html
12
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-or-later
13
/****************************************************************************/
14
/// @file GUIBaseVehicle.cpp
15
/// @author Daniel Krajzewicz
16
/// @author Jakob Erdmann
17
/// @author Michael Behrisch
18
/// @author Laura Bieker-Walz
19
/// @date Sept 2002
20
///
21
// A MSVehicle extended by some values for usage within the gui
22
/****************************************************************************/
23
#include <config.h>
24
25
#include <cmath>
26
#include <vector>
27
#include <string>
28
#include <functional>
29
#include <utils/common/MsgHandler.h>
30
#include <utils/common/StringUtils.h>
31
#include <utils/common/StringTokenizer.h>
32
#include <utils/geom/GeomHelper.h>
33
#include <utils/vehicle/SUMOVehicleParameter.h>
34
#include <utils/emissions/PollutantsInterface.h>
35
#include <utils/gui/globjects/GLIncludes.h>
36
#include <utils/gui/windows/GUISUMOAbstractView.h>
37
#include <utils/gui/windows/GUIAppEnum.h>
38
#include <utils/gui/images/GUITexturesHelper.h>
39
#include <utils/gui/div/GUIGlobalSelection.h>
40
#include <utils/gui/div/GLHelper.h>
41
#include <utils/gui/div/GUIGlobalSelection.h>
42
#include <utils/gui/div/GUIBaseVehicleHelper.h>
43
#include <mesosim/MEVehicle.h>
44
#include <mesosim/MELoop.h>
45
#include <microsim/MSVehicle.h>
46
#include <microsim/MSLane.h>
47
#include <microsim/logging/CastingFunctionBinding.h>
48
#include <microsim/logging/FunctionBinding.h>
49
#include <microsim/MSVehicleControl.h>
50
#include <microsim/MSStop.h>
51
#include <microsim/MSTrainHelper.h>
52
#include <microsim/lcmodels/MSAbstractLaneChangeModel.h>
53
#include <microsim/devices/MSDevice_Vehroutes.h>
54
#include <microsim/devices/MSDevice_Transportable.h>
55
#include <microsim/devices/MSDevice_BTreceiver.h>
56
#include <microsim/trigger/MSStoppingPlaceRerouter.h>
57
#include <gui/GUIApplicationWindow.h>
58
#include <gui/GUIGlobals.h>
59
#include <utils/gui/div/GUIDesigns.h>
60
61
#include "GUIBaseVehicle.h"
62
#include "GUIChargingStation.h"
63
#include "GUIPerson.h"
64
#include "GUIContainer.h"
65
#include "GUINet.h"
66
#include "GUIEdge.h"
67
#include "GUILane.h"
68
#include "GUIParkingArea.h"
69
70
//#define DRAW_BOUNDING_BOX
71
72
// ===========================================================================
73
// FOX callback mapping
74
// ===========================================================================
75
FXDEFMAP(GUIBaseVehicle::GUIBaseVehiclePopupMenu) GUIBaseVehiclePopupMenuMap[] = {
76
FXMAPFUNC(SEL_COMMAND, MID_SHOW_ALLROUTES, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowAllRoutes),
77
FXMAPFUNC(SEL_COMMAND, MID_HIDE_ALLROUTES, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideAllRoutes),
78
FXMAPFUNC(SEL_COMMAND, MID_SHOW_CURRENTROUTE, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowCurrentRoute),
79
FXMAPFUNC(SEL_COMMAND, MID_HIDE_CURRENTROUTE, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideCurrentRoute),
80
FXMAPFUNC(SEL_COMMAND, MID_SHOW_FUTUREROUTE, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowFutureRoute),
81
FXMAPFUNC(SEL_COMMAND, MID_HIDE_FUTUREROUTE, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideFutureRoute),
82
FXMAPFUNC(SEL_COMMAND, MID_SHOW_ROUTE_NOLOOPS, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowRouteNoLoops),
83
FXMAPFUNC(SEL_COMMAND, MID_HIDE_ROUTE_NOLOOPS, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideRouteNoLoops),
84
FXMAPFUNC(SEL_COMMAND, MID_SHOW_BEST_LANES, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowBestLanes),
85
FXMAPFUNC(SEL_COMMAND, MID_HIDE_BEST_LANES, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideBestLanes),
86
FXMAPFUNC(SEL_COMMAND, MID_START_TRACK, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdStartTrack),
87
FXMAPFUNC(SEL_COMMAND, MID_STOP_TRACK, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdStopTrack),
88
FXMAPFUNC(SEL_COMMAND, MID_SHOW_LFLINKITEMS, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowLFLinkItems),
89
FXMAPFUNC(SEL_COMMAND, MID_HIDE_LFLINKITEMS, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideLFLinkItems),
90
FXMAPFUNC(SEL_COMMAND, MID_SHOW_FOES, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowFoes),
91
FXMAPFUNC(SEL_COMMAND, MID_SELECT_TRANSPORTED, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdSelectTransported),
92
FXMAPFUNC(SEL_COMMAND, MID_REMOVE_OBJECT, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdRemoveObject),
93
FXMAPFUNC(SEL_COMMAND, MID_TOGGLE_STOP, GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdToggleStop),
94
};
95
96
// Object implementation
97
FXIMPLEMENT(GUIBaseVehicle::GUIBaseVehiclePopupMenu, GUIGLObjectPopupMenu, GUIBaseVehiclePopupMenuMap, ARRAYNUMBER(GUIBaseVehiclePopupMenuMap))
98
99
// ===========================================================================
100
// method definitions
101
// ===========================================================================
102
/* -------------------------------------------------------------------------
103
* GUIBaseVehicle::GUIBaseVehiclePopupMenu - methods
104
* ----------------------------------------------------------------------- */
105
GUIBaseVehicle::GUIBaseVehiclePopupMenu::GUIBaseVehiclePopupMenu(
106
GUIMainWindow& app, GUISUMOAbstractView& parent, GUIGlObject* o) :
107
GUIGLObjectPopupMenu(app, parent, o) {
108
}
109
110
111
GUIBaseVehicle::GUIBaseVehiclePopupMenu::~GUIBaseVehiclePopupMenu() {}
112
113
114
long
115
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowAllRoutes(FXObject*, FXSelector, void*) {
116
assert(myObject->getType() == GLO_VEHICLE);
117
if (!static_cast<GUIBaseVehicle*>(myObject)->hasActiveAddVisualisation(myParent, VO_SHOW_ALL_ROUTES)) {
118
static_cast<GUIBaseVehicle*>(myObject)->addActiveAddVisualisation(myParent, VO_SHOW_ALL_ROUTES);
119
}
120
return 1;
121
}
122
123
long
124
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideAllRoutes(FXObject*, FXSelector, void*) {
125
assert(myObject->getType() == GLO_VEHICLE);
126
static_cast<GUIBaseVehicle*>(myObject)->removeActiveAddVisualisation(myParent, VO_SHOW_ALL_ROUTES);
127
return 1;
128
}
129
130
131
long
132
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowCurrentRoute(FXObject*, FXSelector, void*) {
133
assert(myObject->getType() == GLO_VEHICLE);
134
if (!static_cast<GUIBaseVehicle*>(myObject)->hasActiveAddVisualisation(myParent, VO_SHOW_ROUTE)) {
135
static_cast<GUIBaseVehicle*>(myObject)->addActiveAddVisualisation(myParent, VO_SHOW_ROUTE);
136
}
137
return 1;
138
}
139
140
long
141
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideCurrentRoute(FXObject*, FXSelector, void*) {
142
assert(myObject->getType() == GLO_VEHICLE);
143
static_cast<GUIBaseVehicle*>(myObject)->removeActiveAddVisualisation(myParent, VO_SHOW_ROUTE);
144
return 1;
145
}
146
147
148
long
149
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowFutureRoute(FXObject*, FXSelector, void*) {
150
assert(myObject->getType() == GLO_VEHICLE);
151
if (!static_cast<GUIBaseVehicle*>(myObject)->hasActiveAddVisualisation(myParent, VO_SHOW_FUTURE_ROUTE)) {
152
static_cast<GUIBaseVehicle*>(myObject)->addActiveAddVisualisation(myParent, VO_SHOW_FUTURE_ROUTE);
153
}
154
return 1;
155
}
156
157
long
158
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideFutureRoute(FXObject*, FXSelector, void*) {
159
assert(myObject->getType() == GLO_VEHICLE);
160
static_cast<GUIBaseVehicle*>(myObject)->removeActiveAddVisualisation(myParent, VO_SHOW_FUTURE_ROUTE);
161
return 1;
162
}
163
164
165
long
166
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowRouteNoLoops(FXObject*, FXSelector, void*) {
167
assert(myObject->getType() == GLO_VEHICLE);
168
if (!static_cast<GUIBaseVehicle*>(myObject)->hasActiveAddVisualisation(myParent, VO_SHOW_ROUTE_NOLOOP)) {
169
static_cast<GUIBaseVehicle*>(myObject)->addActiveAddVisualisation(myParent, VO_SHOW_ROUTE_NOLOOP);
170
}
171
return 1;
172
}
173
174
long
175
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideRouteNoLoops(FXObject*, FXSelector, void*) {
176
assert(myObject->getType() == GLO_VEHICLE);
177
static_cast<GUIBaseVehicle*>(myObject)->removeActiveAddVisualisation(myParent, VO_SHOW_ROUTE_NOLOOP);
178
return 1;
179
}
180
181
182
long
183
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowBestLanes(FXObject*, FXSelector, void*) {
184
assert(myObject->getType() == GLO_VEHICLE);
185
if (!static_cast<GUIBaseVehicle*>(myObject)->hasActiveAddVisualisation(myParent, VO_SHOW_BEST_LANES)) {
186
static_cast<GUIBaseVehicle*>(myObject)->addActiveAddVisualisation(myParent, VO_SHOW_BEST_LANES);
187
}
188
return 1;
189
}
190
191
long
192
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideBestLanes(FXObject*, FXSelector, void*) {
193
assert(myObject->getType() == GLO_VEHICLE);
194
static_cast<GUIBaseVehicle*>(myObject)->removeActiveAddVisualisation(myParent, VO_SHOW_BEST_LANES);
195
return 1;
196
}
197
198
199
long
200
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdStartTrack(FXObject*, FXSelector, void*) {
201
assert(myObject->getType() == GLO_VEHICLE);
202
if (myParent->getTrackedID() != static_cast<GUIBaseVehicle*>(myObject)->getGlID()) {
203
myParent->startTrack(static_cast<GUIBaseVehicle*>(myObject)->getGlID());
204
}
205
return 1;
206
}
207
208
long
209
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdStopTrack(FXObject*, FXSelector, void*) {
210
assert(myObject->getType() == GLO_VEHICLE);
211
myParent->stopTrack();
212
return 1;
213
}
214
215
216
long
217
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowLFLinkItems(FXObject*, FXSelector, void*) {
218
assert(myObject->getType() == GLO_VEHICLE);
219
if (!static_cast<GUIBaseVehicle*>(myObject)->hasActiveAddVisualisation(myParent, VO_SHOW_LFLINKITEMS)) {
220
static_cast<GUIBaseVehicle*>(myObject)->addActiveAddVisualisation(myParent, VO_SHOW_LFLINKITEMS);
221
}
222
return 1;
223
}
224
225
long
226
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdHideLFLinkItems(FXObject*, FXSelector, void*) {
227
assert(myObject->getType() == GLO_VEHICLE);
228
static_cast<GUIBaseVehicle*>(myObject)->removeActiveAddVisualisation(myParent, VO_SHOW_LFLINKITEMS);
229
return 1;
230
}
231
232
long
233
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdShowFoes(FXObject*, FXSelector, void*) {
234
assert(myObject->getType() == GLO_VEHICLE);
235
static_cast<GUIBaseVehicle*>(myObject)->selectBlockingFoes();
236
myParent->update();
237
return 1;
238
}
239
240
241
long
242
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdSelectTransported(FXObject*, FXSelector, void*) {
243
assert(myObject->getType() == GLO_VEHICLE);
244
const MSBaseVehicle& veh = static_cast<GUIBaseVehicle*>(myObject)->getVehicle();
245
for (const MSTransportable* t : veh.getPersons()) {
246
gSelected.select((static_cast<const GUIPerson*>(t))->getGlID());
247
}
248
for (MSTransportable* t : veh.getContainers()) {
249
gSelected.select((static_cast<const GUIContainer*>(t))->getGlID());
250
}
251
myParent->update();
252
return 1;
253
}
254
255
long
256
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdRemoveObject(FXObject*, FXSelector, void*) {
257
GUIBaseVehicle* baseVeh = static_cast<GUIBaseVehicle*>(myObject);
258
MSVehicle* microVeh = dynamic_cast<MSVehicle*>(&baseVeh->myVehicle);
259
if (microVeh != nullptr) {
260
MSLane* lane = microVeh->getMutableLane();
261
if (lane != nullptr) {
262
lane->getVehiclesSecure();
263
lane->removeVehicle(microVeh, MSMoveReminder::NOTIFICATION_VAPORIZED_GUI);
264
}
265
microVeh->onRemovalFromNet(MSMoveReminder::NOTIFICATION_VAPORIZED_GUI);
266
if (lane != nullptr) {
267
lane->releaseVehicles();
268
}
269
} else {
270
MEVehicle* mesoVeh = dynamic_cast<MEVehicle*>(&baseVeh->myVehicle);
271
MSGlobals::gMesoNet->vaporizeCar(mesoVeh, MSMoveReminder::NOTIFICATION_VAPORIZED_GUI);
272
}
273
MSNet::getInstance()->getVehicleControl().scheduleVehicleRemoval(&baseVeh->myVehicle);
274
myParent->destroyPopup();
275
myParent->update();
276
return 1;
277
}
278
279
280
long
281
GUIBaseVehicle::GUIBaseVehiclePopupMenu::onCmdToggleStop(FXObject*, FXSelector, void*) {
282
GUIBaseVehicle* baseVeh = static_cast<GUIBaseVehicle*>(myObject);
283
MSVehicle* microVeh = dynamic_cast<MSVehicle*>(&baseVeh->myVehicle);
284
if (microVeh != nullptr) {
285
if (microVeh->isStopped()) {
286
microVeh->resumeFromStopping();
287
} else {
288
std::string errorOut;
289
const double brakeGap = microVeh->getCarFollowModel().brakeGap(microVeh->getSpeed());
290
std::pair<const MSLane*, double> stopPos = microVeh->getLanePosAfterDist(brakeGap);
291
if (stopPos.first != nullptr) {
292
SUMOVehicleParameter::Stop stop;
293
stop.lane = stopPos.first->getID();
294
stop.startPos = stopPos.second;
295
stop.endPos = stopPos.second + POSITION_EPS;
296
stop.duration = TIME2STEPS(3600);
297
microVeh->addTraciStop(stop, errorOut);
298
if (errorOut != "") {
299
WRITE_WARNING(errorOut);
300
}
301
}
302
}
303
} else {
304
WRITE_WARNING(TL("GUI-triggered stop not implemented for meso"));
305
}
306
myParent->update();
307
return 1;
308
}
309
310
311
/* -------------------------------------------------------------------------
312
* GUIBaseVehicle - methods
313
* ----------------------------------------------------------------------- */
314
315
GUIBaseVehicle::GUIBaseVehicle(MSBaseVehicle& vehicle) :
316
GUIGlObject(GLO_VEHICLE, vehicle.getID(), GUIIconSubSys::getIcon(GUIIcon::VEHICLE)),
317
myVehicle(vehicle),
318
myPopup(nullptr) {
319
// as it is possible to show all vehicle routes, we have to store them... (bug [ 2519761 ])
320
myRoutes = MSDevice_Vehroutes::buildVehicleDevices(myVehicle, myVehicle.myDevices, 5);
321
myVehicle.myMoveReminders.push_back(std::make_pair(myRoutes, 0.));
322
mySeatPositions.push_back(Seat()); // ensure length 1
323
myContainerPositions.push_back(Seat()); // ensure length 1
324
}
325
326
327
GUIBaseVehicle::~GUIBaseVehicle() {
328
myLock.lock();
329
for (std::map<GUISUMOAbstractView*, int>::iterator i = myAdditionalVisualizations.begin(); i != myAdditionalVisualizations.end(); ++i) {
330
if (i->first->getTrackedID() == getGlID()) {
331
i->first->stopTrack();
332
}
333
while (i->first->removeAdditionalGLVisualisation(this));
334
}
335
myLock.unlock();
336
delete myRoutes;
337
if (myPopup != nullptr) {
338
myPopup->getParentView()->destroyPopup();
339
}
340
}
341
342
343
GUIGLObjectPopupMenu*
344
GUIBaseVehicle::getPopUpMenu(GUIMainWindow& app, GUISUMOAbstractView& parent) {
345
GUIGLObjectPopupMenu* ret = new GUIBaseVehiclePopupMenu(app, parent, this);
346
buildPopupHeader(ret, app);
347
buildCenterPopupEntry(ret);
348
buildNameCopyPopupEntry(ret);
349
buildSelectionPopupEntry(ret);
350
//
351
if (hasActiveAddVisualisation(&parent, VO_SHOW_ROUTE)) {
352
GUIDesigns::buildFXMenuCommand(ret, TL("Hide Current Route"), nullptr, ret, MID_HIDE_CURRENTROUTE);
353
} else {
354
GUIDesigns::buildFXMenuCommand(ret, TL("Show Current Route"), nullptr, ret, MID_SHOW_CURRENTROUTE);
355
}
356
if (hasActiveAddVisualisation(&parent, VO_SHOW_FUTURE_ROUTE)) {
357
GUIDesigns::buildFXMenuCommand(ret, TL("Hide Future Route"), nullptr, ret, MID_HIDE_FUTUREROUTE);
358
} else {
359
GUIDesigns::buildFXMenuCommand(ret, TL("Show Future Route"), nullptr, ret, MID_SHOW_FUTUREROUTE);
360
}
361
if (hasActiveAddVisualisation(&parent, VO_SHOW_ALL_ROUTES)) {
362
GUIDesigns::buildFXMenuCommand(ret, TL("Hide All Routes"), nullptr, ret, MID_HIDE_ALLROUTES);
363
} else {
364
GUIDesigns::buildFXMenuCommand(ret, TL("Show All Routes"), nullptr, ret, MID_SHOW_ALLROUTES);
365
}
366
if (hasActiveAddVisualisation(&parent, VO_SHOW_ROUTE_NOLOOP)) {
367
FXMenuCheck* showLoops = new FXMenuCheck(ret, TL("Draw looped routes"), ret, MID_HIDE_ROUTE_NOLOOPS);
368
showLoops->setCheck(false);
369
} else {
370
FXMenuCheck* showLoops = new FXMenuCheck(ret, TL("Draw looped routes"), ret, MID_SHOW_ROUTE_NOLOOPS);
371
showLoops->setCheck(true);
372
}
373
if (hasActiveAddVisualisation(&parent, VO_SHOW_BEST_LANES)) {
374
GUIDesigns::buildFXMenuCommand(ret, TL("Hide Best Lanes"), nullptr, ret, MID_HIDE_BEST_LANES);
375
} else {
376
GUIDesigns::buildFXMenuCommand(ret, TL("Show Best Lanes"), nullptr, ret, MID_SHOW_BEST_LANES);
377
}
378
if (hasActiveAddVisualisation(&parent, VO_SHOW_LFLINKITEMS)) {
379
GUIDesigns::buildFXMenuCommand(ret, TL("Hide Link Items"), nullptr, ret, MID_HIDE_LFLINKITEMS);
380
} else {
381
GUIDesigns::buildFXMenuCommand(ret, TL("Show Link Items"), nullptr, ret, MID_SHOW_LFLINKITEMS);
382
}
383
new FXMenuSeparator(ret);
384
if (parent.getTrackedID() != getGlID()) {
385
GUIDesigns::buildFXMenuCommand(ret, TL("Start Tracking"), nullptr, ret, MID_START_TRACK);
386
} else {
387
GUIDesigns::buildFXMenuCommand(ret, TL("Stop Tracking"), nullptr, ret, MID_STOP_TRACK);
388
}
389
GUIDesigns::buildFXMenuCommand(ret, TL("Select Foes"), nullptr, ret, MID_SHOW_FOES);
390
if (myVehicle.getPersons().size() + myVehicle.getContainers().size() > 0) {
391
GUIDesigns::buildFXMenuCommand(ret, TL("Select transported"), nullptr, ret, MID_SELECT_TRANSPORTED);
392
}
393
GUIDesigns::buildFXMenuCommand(ret, myVehicle.isStopped() ? TL("Abort stop") : TL("Stop"), nullptr, ret, MID_TOGGLE_STOP);
394
GUIDesigns::buildFXMenuCommand(ret, TL("Remove"), nullptr, ret, MID_REMOVE_OBJECT);
395
396
new FXMenuSeparator(ret);
397
//
398
buildShowParamsPopupEntry(ret, false);
399
buildShowTypeParamsPopupEntry(ret);
400
buildPositionCopyEntry(ret, app);
401
myPopup = ret;
402
return ret;
403
}
404
405
406
void
407
GUIBaseVehicle::removedPopupMenu() {
408
myPopup = nullptr;
409
}
410
411
412
double
413
GUIBaseVehicle::getExaggeration(const GUIVisualizationSettings& s) const {
414
return (s.vehicleSize.getExaggeration(s, this) *
415
s.vehicleScaler.getScheme().getColor(getScaleValue(s, s.vehicleScaler.getActive())));
416
}
417
418
419
Boundary
420
GUIBaseVehicle::getCenteringBoundary() const {
421
Boundary b;
422
b.add(getVisualPosition(GUIGlobals::gSecondaryShape));
423
b.grow(myVehicle.getVehicleType().getLength());
424
return b;
425
}
426
427
428
const std::string
429
GUIBaseVehicle::getOptionalName() const {
430
return myVehicle.getParameter().getParameter("name", "");
431
}
432
433
434
void
435
GUIBaseVehicle::drawOnPos(const GUIVisualizationSettings& s, const Position& pos, const double angle) const {
436
GLHelper::pushName(getGlID());
437
GLHelper::pushMatrix();
438
Position p1 = pos;
439
const double degAngle = RAD2DEG(angle + M_PI / 2.);
440
const double length = getVType().getLength();
441
if (s.trueZ) {
442
glTranslated(p1.x(), p1.y(), p1.z() + 1);
443
} else {
444
glTranslated(p1.x(), p1.y(), getType());
445
}
446
glRotated(degAngle, 0, 0, 1);
447
RGBColor col = setColor(s);
448
// scale
449
const double upscale = getExaggeration(s);
450
const bool s2 = s.secondaryShape;
451
452
if (upscale > 1 && s.laneWidthExaggeration > 1 && myVehicle.isOnRoad()) {
453
// optionally shift according to edge exaggeration
454
double offsetFromLeftBorder = myVehicle.getCurrentEdge()->getWidth() - myVehicle.getRightSideOnEdge() - myVehicle.getVehicleType().getWidth() / 2;
455
glTranslated((s.laneWidthExaggeration - 1) * -offsetFromLeftBorder / 2, 0, 0);
456
}
457
458
double upscaleLength = MSTrainHelper::getUpscaleLength(upscale, length, getVType().getWidth(), s.vehicleQuality);
459
glScaled(upscale, upscaleLength, 1);
460
/*
461
MSLaneChangeModel::DK2004 &m2 = static_cast<MSLaneChangeModel::DK2004&>(veh->getLaneChangeModel());
462
if((m2.getState()&LCA_URGENT)!=0) {
463
glColor3d(1, .4, .4);
464
} else if((m2.getState()&LCA_SPEEDGAIN)!=0) {
465
glColor3d(.4, .4, 1);
466
} else {
467
glColor3d(.4, 1, .4);
468
}
469
*/
470
// draw the vehicle
471
bool drawCarriages = false;
472
const double geometryFactor = (s.scaleLength ?
473
((myVehicle.getLane() != nullptr
474
? myVehicle.getLane()->getLengthGeometryFactor(s2)
475
: (myVehicle.getEdge()->getLanes().size() > 0 ? myVehicle.getEdge()->getLanes()[0]->getLengthGeometryFactor(s2) : 1)))
476
: 1);
477
double scaledLength = length * geometryFactor;
478
if (col.alpha() != 0) {
479
switch (s.vehicleQuality) {
480
case 0:
481
GUIBaseVehicleHelper::drawAction_drawVehicleAsTrianglePlus(getVType().getWidth(), scaledLength, drawReversed(s));
482
break;
483
case 1:
484
GUIBaseVehicleHelper::drawAction_drawVehicleAsBoxPlus(getVType().getWidth(), scaledLength, drawReversed(s));
485
break;
486
case 2:
487
drawCarriages = drawAction_drawVehicleAsPolyWithCarriagges(s, scaledLength);
488
// draw flashing blue light for emergency vehicles
489
if (getVType().getGuiShape() == SUMOVehicleShape::EMERGENCY) {
490
glTranslated(0, 0, .1);
491
drawAction_drawVehicleBlueLight();
492
}
493
break;
494
case 3:
495
drawCarriages = drawAction_drawVehicleAsPolyWithCarriagges(s, scaledLength, true);
496
break;
497
case 4: {
498
// do not scale circle radius by lengthGeometryFactor nor length and reduce the effect of width
499
const double w = 1.8 * sqrt(getVType().getWidth() / 1.8);
500
GUIBaseVehicleHelper::drawAction_drawVehicleAsCircle(w, s.scale * upscale);
501
// display text at circle center
502
scaledLength = 0;
503
break;
504
}
505
default:
506
break;
507
}
508
if (s.drawMinGap) {
509
const double minGap = -getVType().getMinGap();
510
glColor3d(0., 1., 0.);
511
glBegin(GL_LINES);
512
glVertex2d(0., 0);
513
glVertex2d(0., minGap);
514
glVertex2d(-.5, minGap);
515
glVertex2d(.5, minGap);
516
glEnd();
517
}
518
if (s.drawBrakeGap && !MSGlobals::gUseMesoSim
519
&& (!s.vehicleSize.constantSizeSelected || myVehicle.isSelected())) {
520
const double brakeGap = -static_cast<MSVehicle&>(myVehicle).getCarFollowModel().brakeGap(myVehicle.getSpeed());
521
glColor3d(1., 0., 0.);
522
glBegin(GL_LINES);
523
glVertex2d(0., 0);
524
glVertex2d(0., brakeGap);
525
glVertex2d(-.5, brakeGap);
526
glVertex2d(.5, brakeGap);
527
glEnd();
528
}
529
if (s.showBTRange) {
530
MSVehicleDevice_BTreceiver* dev = static_cast<MSVehicleDevice_BTreceiver*>(myVehicle.getDevice(typeid(MSVehicleDevice_BTreceiver)));
531
if (dev != nullptr) {
532
glColor3d(1., 0., 0.);
533
GLHelper::drawOutlineCircle(dev->getRange(), dev->getRange() - .2, 32);
534
}
535
}
536
// draw the blinker and brakelights if wished
537
if (s.showBlinker) {
538
glTranslated(0, 0, .1);
539
switch (getVType().getGuiShape()) {
540
case SUMOVehicleShape::PEDESTRIAN:
541
case SUMOVehicleShape::BICYCLE:
542
case SUMOVehicleShape::SCOOTER:
543
case SUMOVehicleShape::ANT:
544
case SUMOVehicleShape::SHIP:
545
case SUMOVehicleShape::RAIL:
546
case SUMOVehicleShape::RAIL_CARGO:
547
case SUMOVehicleShape::RAIL_CAR:
548
case SUMOVehicleShape::AIRCRAFT:
549
break;
550
case SUMOVehicleShape::MOTORCYCLE:
551
case SUMOVehicleShape::MOPED:
552
drawAction_drawVehicleBlinker(scaledLength);
553
drawAction_drawVehicleBrakeLight(scaledLength, true);
554
break;
555
default:
556
// only SUMOVehicleShape::RAIL_CAR has blinkers and brake lights but they are drawn along with the carriages
557
if (!drawCarriages) {
558
drawAction_drawVehicleBlinker(scaledLength);
559
drawAction_drawVehicleBrakeLight(scaledLength);
560
}
561
break;
562
}
563
}
564
// draw the wish to change the lane
565
if (s.drawLaneChangePreference) {
566
/*
567
if(gSelected.isSelected(GLO_VEHICLE, veh->getGlID())) {
568
MSLaneChangeModel::DK2004 &m = static_cast<MSLaneChangeModel::DK2004&>(veh->getLaneChangeModel());
569
glColor3d(.5, .5, 1);
570
glBegin(GL_LINES);
571
glVertex2f(0, 0);
572
glVertex2f(m.getChangeProbability(), .5);
573
glEnd();
574
575
glColor3d(1, 0, 0);
576
glBegin(GL_LINES);
577
glVertex2f(0.1, 0);
578
glVertex2f(0.1, m.myMaxJam1);
579
glEnd();
580
581
glColor3d(0, 1, 0);
582
glBegin(GL_LINES);
583
glVertex2f(-0.1, 0);
584
glVertex2f(-0.1, m.myTDist);
585
glEnd();
586
}
587
*/
588
}
589
}
590
glTranslated(0, MIN2(scaledLength / 2, double(5)), -getType()); // drawing name at GLO_MAX fails unless translating z
591
glScaled(1 / upscale, 1 / upscaleLength, 1);
592
glRotated(-degAngle, 0, 0, 1);
593
drawName(Position(0, 0), s.scale, s.vehicleName, s.angle);
594
if (s.vehicleName.show(this) && myVehicle.getParameter().line != "") {
595
glRotated(-s.angle, 0, 0, 1);
596
glTranslated(0, 0.7 * s.vehicleName.scaledSize(s.scale), 0);
597
glRotated(s.angle, 0, 0, 1);
598
GLHelper::drawTextSettings(s.vehicleName, "line:" + myVehicle.getParameter().line, Position(0, 0), s.scale, s.angle);
599
}
600
if (s.vehicleValue.show(this)) {
601
glRotated(-s.angle, 0, 0, 1);
602
glTranslated(0, 0.7 * s.vehicleName.scaledSize(s.scale), 0);
603
glRotated(s.angle, 0, 0, 1);
604
const double value = getColorValue(s, s.vehicleColorer.getActive());
605
if (value != s.MISSING_DATA) {
606
GLHelper::drawTextSettings(s.vehicleValue, toString(value), Position(0, 0), s.scale, s.angle);
607
}
608
}
609
if (s.vehicleScaleValue.show(this)) {
610
glRotated(-s.angle, 0, 0, 1);
611
glTranslated(0, 0.7 * s.vehicleName.scaledSize(s.scale), 0);
612
glRotated(s.angle, 0, 0, 1);
613
const double value = getScaleValue(s, s.vehicleScaler.getActive());
614
if (value != s.MISSING_DATA) {
615
GLHelper::drawTextSettings(s.vehicleScaleValue, toString(value), Position(0, 0), s.scale, s.angle);
616
}
617
}
618
if (s.vehicleText.show(this)) {
619
std::string error;
620
std::string value = myVehicle.getPrefixedParameter(s.vehicleTextParam, error);
621
if (value != "") {
622
auto lines = StringTokenizer(value, StringTokenizer::NEWLINE).getVector();
623
glRotated(-s.angle, 0, 0, 1);
624
glTranslated(0, 0.7 * s.vehicleText.scaledSize(s.scale) * (double)lines.size(), 0);
625
glRotated(s.angle, 0, 0, 1);
626
for (std::string& line : lines) {
627
GLHelper::drawTextSettings(s.vehicleText, line, Position(0, 0), s.scale, s.angle);
628
glRotated(-s.angle, 0, 0, 1);
629
glTranslated(0, -0.7 * s.vehicleText.scaledSize(s.scale), 0);
630
glRotated(s.angle, 0, 0, 1);
631
}
632
}
633
}
634
if (s.showParkingInfo && myAdditionalVisualizations.size() != 0 && hasActiveAddVisualisation(
635
myAdditionalVisualizations.begin()->first, VO_SHOW_ROUTE | VO_SHOW_FUTURE_ROUTE | VO_SHOW_ALL_ROUTES)) {
636
glRotated(-s.angle, 0, 0, 1);
637
glTranslated(0, 0.7 * s.vehicleName.scaledSize(s.scale), 0);
638
glRotated(s.angle, 0, 0, 1);
639
const double value = myVehicle.getNumberParkingReroutes();
640
GLHelper::drawTextSettings(s.vehicleName, toString(value), Position(0, 0), s.scale, s.angle);
641
}
642
643
if (!drawCarriages) {
644
mySeatPositions.clear();
645
myContainerPositions.clear();
646
int requiredSeats = getNumPassengers();
647
int requiredContainerPositions = getNumContainers();
648
const Position back = (p1 + Position(-scaledLength * upscaleLength, 0)).rotateAround2D(angle, p1);
649
double extraOffset = scaledLength * 0.15;
650
computeSeats(p1, back, SUMO_const_waitingPersonWidth, getVType().getPersonCapacity(), upscale, requiredSeats, mySeatPositions, extraOffset);
651
computeSeats(p1, back, SUMO_const_waitingContainerWidth, getVType().getContainerCapacity(), upscale, requiredContainerPositions, myContainerPositions, extraOffset);
652
}
653
654
GLHelper::popMatrix();
655
GLHelper::popName();
656
drawAction_drawPersonsAndContainers(s);
657
}
658
659
660
void
661
GUIBaseVehicle::drawGL(const GUIVisualizationSettings& s) const {
662
drawOnPos(s, getVisualPosition(s.secondaryShape), getVisualAngle(s.secondaryShape));
663
}
664
665
666
void
667
GUIBaseVehicle::drawGLAdditional(GUISUMOAbstractView* const parent, const GUIVisualizationSettings& s) const {
668
if (!myVehicle.isOnRoad()) {
669
drawGL(s);
670
}
671
GLHelper::pushName(getGlID());
672
GLHelper::pushMatrix();
673
glTranslated(0, 0, getType() - .1); // don't draw on top of other cars
674
if (hasActiveAddVisualisation(parent, VO_SHOW_BEST_LANES)) {
675
drawBestLanes();
676
}
677
bool noLoop = hasActiveAddVisualisation(parent, VO_SHOW_ROUTE_NOLOOP);
678
if (hasActiveAddVisualisation(parent, VO_SHOW_ROUTE)) {
679
drawRoute(s, 0, 0.25, false, noLoop);
680
}
681
if (hasActiveAddVisualisation(parent, VO_SHOW_FUTURE_ROUTE)) {
682
drawRoute(s, 0, 0.25, true, noLoop);
683
}
684
if (hasActiveAddVisualisation(parent, VO_SHOW_ALL_ROUTES)) {
685
if (myVehicle.getNumberReroutes() > 0) {
686
const int noReroutePlus1 = myVehicle.getNumberReroutes() + 1;
687
for (int i = noReroutePlus1 - 1; i >= 0; i--) {
688
double darken = double(0.4) / double(noReroutePlus1) * double(i);
689
drawRoute(s, i, darken);
690
}
691
} else {
692
drawRoute(s, 0, 0.25, false, noLoop);
693
}
694
}
695
if (hasActiveAddVisualisation(parent, VO_SHOW_LFLINKITEMS)) {
696
drawAction_drawLinkItems(s);
697
}
698
GLHelper::popMatrix();
699
GLHelper::popName();
700
}
701
702
703
void
704
GUIBaseVehicle::drawLinkItem(const Position& pos, SUMOTime arrivalTime, SUMOTime leaveTime, double exagerate) {
705
glTranslated(pos.x(), pos.y(), -.1);
706
GLHelper::drawFilledCircle(1);
707
std::string times = toString(STEPS2TIME(arrivalTime)) + "/" + toString(STEPS2TIME(leaveTime));
708
GLHelper::drawText(times.c_str(), Position(), .1, 1.6 * exagerate, RGBColor::GREEN, 0);
709
glTranslated(-pos.x(), -pos.y(), .1);
710
}
711
712
713
RGBColor
714
GUIBaseVehicle::setColor(const GUIVisualizationSettings& s) const {
715
RGBColor col;
716
const GUIColorer& c = s.vehicleColorer;
717
if (!setFunctionalColor(c.getActive(), &myVehicle, col)) {
718
col = c.getScheme().getColor(getColorValue(s, c.getActive()));
719
}
720
GLHelper::setColor(col);
721
return col;
722
}
723
724
725
bool
726
GUIBaseVehicle::setFunctionalColor(int activeScheme, const MSBaseVehicle* veh, RGBColor& col) {
727
switch (activeScheme) {
728
case 0: {
729
//test for emergency vehicle
730
if (veh->getVehicleType().getGuiShape() == SUMOVehicleShape::EMERGENCY) {
731
col = RGBColor::WHITE;
732
return true;
733
}
734
//test for firebrigade
735
if (veh->getVehicleType().getGuiShape() == SUMOVehicleShape::FIREBRIGADE) {
736
col = RGBColor::RED;
737
return true;
738
}
739
//test for police car
740
if (veh->getVehicleType().getGuiShape() == SUMOVehicleShape::POLICE) {
741
col = RGBColor::BLUE;
742
return true;
743
}
744
if (veh->getParameter().wasSet(VEHPARS_COLOR_SET)) {
745
col = veh->getParameter().color;
746
return true;
747
}
748
if (veh->getVehicleType().wasSet(VTYPEPARS_COLOR_SET)) {
749
col = veh->getVehicleType().getColor();
750
return true;
751
}
752
if (&(veh->getRoute().getColor()) != &RGBColor::DEFAULT_COLOR) {
753
col = veh->getRoute().getColor();
754
return true;
755
}
756
return false;
757
}
758
case 2: {
759
if (veh->getParameter().wasSet(VEHPARS_COLOR_SET)) {
760
col = veh->getParameter().color;
761
return true;
762
}
763
return false;
764
}
765
case 3: {
766
if (veh->getVehicleType().wasSet(VTYPEPARS_COLOR_SET)) {
767
col = veh->getVehicleType().getColor();
768
return true;
769
}
770
return false;
771
}
772
case 4: {
773
if (&(veh->getRoute().getColor()) != &RGBColor::DEFAULT_COLOR) {
774
col = veh->getRoute().getColor();
775
return true;
776
}
777
return false;
778
}
779
case 5: {
780
Position p = veh->getRoute().getEdges()[0]->getLanes()[0]->getShape()[0];
781
const Boundary& b = ((GUINet*) MSNet::getInstance())->getBoundary();
782
Position center = b.getCenter();
783
double hue = 180. + atan2(center.x() - p.x(), center.y() - p.y()) * 180. / M_PI;
784
double sat = p.distanceTo(center) / center.distanceTo(Position(b.xmin(), b.ymin()));
785
col = RGBColor::fromHSV(hue, sat, 1.);
786
return true;
787
}
788
case 6: {
789
Position p = veh->getRoute().getEdges().back()->getLanes()[0]->getShape()[-1];
790
const Boundary& b = ((GUINet*) MSNet::getInstance())->getBoundary();
791
Position center = b.getCenter();
792
double hue = 180. + atan2(center.x() - p.x(), center.y() - p.y()) * 180. / M_PI;
793
double sat = p.distanceTo(center) / center.distanceTo(Position(b.xmin(), b.ymin()));
794
col = RGBColor::fromHSV(hue, sat, 1.);
795
return true;
796
}
797
case 7: {
798
Position pb = veh->getRoute().getEdges()[0]->getLanes()[0]->getShape()[0];
799
Position pe = veh->getRoute().getEdges().back()->getLanes()[0]->getShape()[-1];
800
const Boundary& b = ((GUINet*) MSNet::getInstance())->getBoundary();
801
double hue = 180. + atan2(pb.x() - pe.x(), pb.y() - pe.y()) * 180. / M_PI;
802
Position minp(b.xmin(), b.ymin());
803
Position maxp(b.xmax(), b.ymax());
804
double sat = pb.distanceTo(pe) / minp.distanceTo(maxp);
805
col = RGBColor::fromHSV(hue, sat, 1.);
806
return true;
807
}
808
case 35: { // color randomly (by pointer hash)
809
std::hash<const MSBaseVehicle*> ptr_hash;
810
const double hue = (double)(ptr_hash(veh) % 360); // [0-360]
811
const double sat = (double)((ptr_hash(veh) / 360) % 67) / 100.0 + 0.33; // [0.33-1]
812
col = RGBColor::fromHSV(hue, sat, 1.);
813
return true;
814
}
815
case 36: { // color by angle
816
double hue = GeomHelper::naviDegree(veh->getAngle());
817
col = RGBColor::fromHSV(hue, 1., 1.);
818
return true;
819
}
820
}
821
return false;
822
}
823
824
825
double
826
GUIBaseVehicle::getScaleValue(const GUIVisualizationSettings& s, int activeScheme) const {
827
switch (activeScheme) {
828
case 0: // uniform
829
return 0;
830
case 1: // selection
831
return myVehicle.isSelected();
832
case 2: // by speed
833
if (myVehicle.isStopped()) {
834
return myVehicle.isParking() ? -2 : -1;
835
}
836
return myVehicle.getSpeed();
837
case 3:
838
return myVehicle.getWaitingSeconds();
839
case 4: {
840
MSVehicle* microVeh = dynamic_cast<MSVehicle*>(&myVehicle);
841
return (microVeh != nullptr ? microVeh->getAccumulatedWaitingSeconds() : 0);
842
}
843
case 5: {
844
MSVehicle* microVeh = dynamic_cast<MSVehicle*>(&myVehicle);
845
return (microVeh != nullptr ? microVeh->getLane()->getVehicleMaxSpeed(microVeh) : myVehicle.getEdge()->getVehicleMaxSpeed(&myVehicle));
846
}
847
case 6:
848
return myVehicle.getNumberReroutes();
849
case 7: {
850
MSVehicle* microVeh = dynamic_cast<MSVehicle*>(&myVehicle);
851
return (microVeh != nullptr
852
? (microVeh->getLaneChangeModel().isOpposite() ? -100 : microVeh->getBestLaneOffset())
853
: 0);
854
}
855
case 8:
856
return myVehicle.getAcceleration();
857
case 9: {
858
MSVehicle* microVeh = dynamic_cast<MSVehicle*>(&myVehicle);
859
return (microVeh != nullptr ? microVeh->getTimeGapOnLane() : 0);
860
}
861
case 10:
862
return STEPS2TIME(myVehicle.getDepartDelay());
863
case 11:
864
return myVehicle.getTimeLossSeconds();
865
case 12:
866
return myVehicle.getStopDelay();
867
case 13:
868
return myVehicle.getStopArrivalDelay();
869
case 14: // by numerical param value
870
std::string error;
871
std::string val = myVehicle.getPrefixedParameter(s.vehicleScaleParam, error);
872
try {
873
if (val == "") {
874
return 0;
875
} else {
876
return StringUtils::toDouble(val);
877
}
878
} catch (NumberFormatException&) {
879
try {
880
return StringUtils::toBool(val);
881
} catch (BoolFormatException&) {
882
WRITE_WARNINGF(TL("Vehicle parameter '%' key '%' is not a number for vehicle '%'."),
883
myVehicle.getParameter().getParameter(s.vehicleScaleParam, "0"), s.vehicleScaleParam, myVehicle.getID());
884
return -1;
885
}
886
}
887
}
888
return 0;
889
}
890
891
892
// ------------ Additional visualisations
893
bool
894
GUIBaseVehicle::hasActiveAddVisualisation(GUISUMOAbstractView* const parent, int which) const {
895
return myAdditionalVisualizations.find(parent) != myAdditionalVisualizations.end() && (myAdditionalVisualizations.find(parent)->second & which) != 0;
896
}
897
898
899
void
900
GUIBaseVehicle::addActiveAddVisualisation(GUISUMOAbstractView* const parent, int which) {
901
if (myAdditionalVisualizations.find(parent) == myAdditionalVisualizations.end()) {
902
myAdditionalVisualizations[parent] = 0;
903
}
904
myAdditionalVisualizations[parent] |= which;
905
if (which != VO_TRACK) {
906
parent->addAdditionalGLVisualisation(this);
907
}
908
}
909
910
911
void
912
GUIBaseVehicle::removeActiveAddVisualisation(GUISUMOAbstractView* const parent, int which) {
913
myAdditionalVisualizations[parent] &= ~which;
914
if (myAdditionalVisualizations[parent] == 0) {
915
myAdditionalVisualizations.erase(parent);
916
}
917
parent->removeAdditionalGLVisualisation(this);
918
}
919
920
921
void
922
GUIBaseVehicle::drawRoute(const GUIVisualizationSettings& s, int routeNo, double darken, bool future, bool noLoop) const {
923
RGBColor vehColor = setColor(s);
924
RGBColor darker = vehColor.changedBrightness((int)(darken * -255));
925
if (darker == RGBColor::BLACK) {
926
darker = vehColor.multiply(1 - darken);
927
}
928
GLHelper::setColor(darker);
929
if (routeNo == 0) {
930
drawRouteHelper(s, myVehicle.getRoutePtr(), future, noLoop, darker);
931
return;
932
}
933
ConstMSRoutePtr route = myRoutes->getRoute(routeNo - 1); // only prior routes are stored
934
if (route != nullptr) {
935
drawRouteHelper(s, route, future, noLoop, darker);
936
}
937
}
938
939
940
void
941
GUIBaseVehicle::drawStopLabels(const GUIVisualizationSettings& s, bool noLoop, const RGBColor& col) const {
942
// (vertical shift for repeated stops at the same position
943
std::map<const MSLane*, int> repeat; // count repeated occurrences of the same position
944
int stopIndex = 0;
945
for (const MSStop& stop : myVehicle.getStops()) {
946
double stopLanePos;
947
if (stop.getSpeed() > 0) {
948
stopLanePos = stop.reached ? stop.pars.endPos : stop.pars.startPos;
949
} else {
950
stopLanePos = stop.reached ? myVehicle.getPositionOnLane() : MAX2(0.0, stop.getEndPos(myVehicle));
951
}
952
if (stop.isOpposite && !stop.reached) {
953
stopLanePos = stop.lane->getLength() - stopLanePos;
954
}
955
Position pos = stop.lane->geometryPositionAtOffset(stopLanePos);
956
GLHelper::setColor(col);
957
GLHelper::drawBoxLines(stop.lane->getShape().getOrthogonal(pos, 10, true, stop.lane->getWidth()), 0.1);
958
std::string label = (stop.getSpeed() > 0
959
? (stop.reached ? "passing waypoint" : "waypoint ")
960
: (stop.reached ? "stopped" : "stop "));
961
if (!stop.reached) {
962
label += toString(stopIndex);
963
}
964
965
if (stop.isOpposite) {
966
label += " (opposite)";
967
}
968
#ifdef _DEBUG
969
label += " (" + toString(stop.edge - myVehicle.getCurrentRouteEdge()) + "e)";
970
#endif
971
if (myVehicle.isStoppedTriggered()) {
972
label += " triggered:";
973
if (stop.triggered) {
974
label += "person";
975
if (stop.numExpectedPerson > 0) {
976
label += "(" + toString(stop.numExpectedPerson) + ")";
977
}
978
}
979
if (stop.containerTriggered) {
980
label += "container";
981
if (stop.numExpectedContainer > 0) {
982
label += "(" + toString(stop.numExpectedContainer) + ")";
983
}
984
}
985
if (stop.joinTriggered) {
986
label += "join";
987
if (stop.pars.join != "") {
988
label += "(" + stop.pars.join + ")";
989
}
990
}
991
}
992
if (stop.pars.ended >= 0 && MSGlobals::gUseStopEnded) {
993
label += " ended:" + time2string(stop.pars.ended);
994
} else if (stop.pars.until >= 0) {
995
label += " until:" + time2string(stop.pars.until);
996
}
997
if (stop.duration >= 0 || stop.pars.duration > 0) {
998
if (STEPS2TIME(stop.duration) > 3600 * 24) {
999
label += " duration:1day+";
1000
} else {
1001
label += " duration:" + time2string(stop.duration);
1002
}
1003
}
1004
if (stop.getSpeed() > 0) {
1005
if (stop.skipOnDemand) {
1006
label += " onDemand (skipped)";
1007
} else {
1008
label += " speed:" + toString(stop.getSpeed());
1009
}
1010
}
1011
if (stop.pars.actType != "") {
1012
label += " actType:" + stop.pars.actType;
1013
}
1014
const double nameSize = s.vehicleName.size / s.scale;
1015
Position pos2 = pos - Position(0, nameSize * repeat[stop.lane]);
1016
if (noLoop && repeat[stop.lane] > 0) {
1017
break;
1018
}
1019
GLHelper::drawTextSettings(s.vehicleText, label, pos2, s.scale, s.angle, 1.0);
1020
repeat[stop.lane]++;
1021
stopIndex++;
1022
}
1023
// indicate arrivalPos if set
1024
if (myVehicle.getParameter().wasSet(VEHPARS_ARRIVALPOS_SET) || myVehicle.getArrivalLane() >= 0) {
1025
const int arrivalEdge = myVehicle.getParameter().arrivalEdge >= 0
1026
? myVehicle.getParameter().arrivalEdge
1027
: (int)myVehicle.getRoute().getEdges().size() - 1;
1028
const MSLane* arrivalLane = myVehicle.getRoute().getEdges()[arrivalEdge]->getLanes()[MAX2(0, myVehicle.getArrivalLane())];
1029
Position pos = arrivalLane->geometryPositionAtOffset(myVehicle.getArrivalPos());
1030
GLHelper::setColor(col);
1031
GLHelper::drawBoxLines(arrivalLane->getShape().getOrthogonal(pos, 10, true, arrivalLane->getWidth() * 0.5, 90), 0.1);
1032
GLHelper::drawBoxLines(arrivalLane->getShape().getOrthogonal(pos, 10, true, arrivalLane->getWidth() * 0.5, 270), 0.1);
1033
GLHelper::drawTextSettings(s.vehicleText, "arrival", pos, s.scale, s.angle, 1.0);
1034
1035
}
1036
}
1037
1038
void
1039
GUIBaseVehicle::drawParkingInfo(const GUIVisualizationSettings& s) const {
1040
if (s.showParkingInfo) {
1041
const StoppingPlaceMemory* pm = myVehicle.getParkingMemory();
1042
if (pm != nullptr) {
1043
for (auto item : *pm) {
1044
const GUIParkingArea* pa = dynamic_cast<const GUIParkingArea*>(item.first);
1045
if (item.second.blockedAtTime >= 0) {
1046
std::string seenAgo = time2string(SIMSTEP - item.second.blockedAtTime);
1047
//if (item.second.blockedAtTime >= 0) {
1048
// seenAgo += ", " + time2string(SIMSTEP - item.second.blockedAtTimeLocal);
1049
//}
1050
GLHelper::drawTextSettings(s.vehicleValue, seenAgo, pa->getSignPos(), s.scale, s.angle, 1.0);
1051
}
1052
if (item.second.score != "") {
1053
const double dist = 0.4 * (s.vehicleText.scaledSize(s.scale) + s.vehicleValue.scaledSize(s.scale));
1054
Position shift(0, -dist);
1055
GLHelper::drawTextSettings(s.vehicleText, item.second.score, pa->getSignPos() + shift, s.scale, s.angle, 1.0);
1056
}
1057
}
1058
}
1059
}
1060
}
1061
1062
void
1063
GUIBaseVehicle::drawChargingInfo(const GUIVisualizationSettings& s) const {
1064
if (s.showChargingInfo) {
1065
const StoppingPlaceMemory* pm = myVehicle.getChargingMemory();
1066
if (pm != nullptr) {
1067
for (auto item : *pm) {
1068
const GUIChargingStation* cs = dynamic_cast<const GUIChargingStation*>(item.first);
1069
if (item.second.blockedAtTime >= 0) {
1070
std::string seenAgo = time2string(SIMSTEP - item.second.blockedAtTime);
1071
GLHelper::drawTextSettings(s.vehicleValue, seenAgo, cs->getSignPos(), s.scale, s.angle, 1.0);
1072
}
1073
if (item.second.score != "") {
1074
const double dist = 0.4 * (s.vehicleText.scaledSize(s.scale) + s.vehicleValue.scaledSize(s.scale));
1075
Position shift(0, -dist);
1076
GLHelper::drawTextSettings(s.vehicleText, item.second.score, cs->getSignPos() + shift, s.scale, s.angle, 1.0);
1077
}
1078
}
1079
}
1080
}
1081
}
1082
1083
const GUIBaseVehicle::Seat&
1084
GUIBaseVehicle::getSeatPosition(int personIndex) const {
1085
/// if there are not enough seats in the vehicle people have to squeeze onto the last seat
1086
return mySeatPositions[MIN2(personIndex, (int)mySeatPositions.size() - 1)];
1087
}
1088
1089
const GUIBaseVehicle::Seat&
1090
GUIBaseVehicle::getContainerPosition(int containerIndex) const {
1091
/// if there are not enough positions in the vehicle containers have to squeeze onto the last position
1092
return myContainerPositions[MIN2(containerIndex, (int)myContainerPositions.size() - 1)];
1093
}
1094
1095
1096
void
1097
GUIBaseVehicle::drawAction_drawPersonsAndContainers(const GUIVisualizationSettings& s) const {
1098
if (myVehicle.myPersonDevice != nullptr) {
1099
const std::vector<MSTransportable*>& ps = myVehicle.myPersonDevice->getTransportables();
1100
int personIndex = 0;
1101
for (std::vector<MSTransportable*>::const_iterator i = ps.begin(); i != ps.end(); ++i) {
1102
GUIPerson* person = dynamic_cast<GUIPerson*>(*i);
1103
assert(person != 0);
1104
person->setPositionInVehicle(getSeatPosition(personIndex++));
1105
person->drawGL(s);
1106
}
1107
}
1108
if (myVehicle.myContainerDevice != nullptr) {
1109
const std::vector<MSTransportable*>& cs = myVehicle.myContainerDevice->getTransportables();
1110
int containerIndex = 0;
1111
for (std::vector<MSTransportable*>::const_iterator i = cs.begin(); i != cs.end(); ++i) {
1112
GUIContainer* container = dynamic_cast<GUIContainer*>(*i);
1113
assert(container != 0);
1114
container->setPositionInVehicle(getContainerPosition(containerIndex++));
1115
container->drawGL(s);
1116
}
1117
}
1118
#ifdef DRAW_BOUNDING_BOX
1119
if (!MSGlobals::gUseMesoSim) {
1120
MSVehicle& microVeh = dynamic_cast<MSVehicle&>(myVehicle);
1121
GLHelper::pushName(getGlID());
1122
GLHelper::pushMatrix();
1123
glTranslated(0, 0, getType());
1124
PositionVector smallBB = microVeh.getBoundingPoly();
1125
glColor3d(0.5, .8, 0);
1126
GLHelper::drawBoxLines(smallBB, 0.3);
1127
glTranslated(0, 0, 0.1);
1128
PositionVector boundingBox = microVeh.getBoundingBox();
1129
boundingBox.push_back(boundingBox.front());
1130
glColor3d(1, 0, 0);
1131
GLHelper::drawBoxLines(boundingBox, 0.15);
1132
GLHelper::popMatrix();
1133
GLHelper::popName();
1134
}
1135
#endif
1136
}
1137
1138
bool
1139
GUIBaseVehicle::drawReversed(const GUIVisualizationSettings& s) const {
1140
return myVehicle.isReversed() && s.drawReversed;
1141
}
1142
1143
bool
1144
GUIBaseVehicle::drawAction_drawVehicleAsPolyWithCarriagges(const GUIVisualizationSettings& s, double scaledLength, bool asImage) const {
1145
if (getVType().getParameter().carriageLength > 0 &&
1146
(!myVehicle.isParking() || myVehicle.getNextStop().parkingarea == nullptr || myVehicle.getNextStop().parkingarea->parkOnRoad())) {
1147
drawAction_drawCarriageClass(s, scaledLength, asImage);
1148
return true;
1149
} else {
1150
if (asImage && GUIBaseVehicleHelper::drawAction_drawVehicleAsImage(
1151
s, getVType().getImgFile(), this, getVType().getWidth(), scaledLength)) {
1152
return false;
1153
}
1154
GUIBaseVehicleHelper::drawAction_drawVehicleAsPoly(s, getVType().getGuiShape(), getVType().getWidth(), scaledLength, -1, myVehicle.isStopped(), drawReversed(s));
1155
return false;
1156
}
1157
}
1158
1159
1160
int
1161
GUIBaseVehicle::getNumPassengers() const {
1162
if (myVehicle.getPersonDevice() != nullptr) {
1163
return (int)myVehicle.getPersonDevice()->size();
1164
}
1165
return 0;
1166
}
1167
1168
1169
int
1170
GUIBaseVehicle::getNumContainers() const {
1171
if (myVehicle.getContainerDevice() != nullptr) {
1172
return (int)myVehicle.getContainerDevice()->size();
1173
}
1174
return 0;
1175
}
1176
1177
std::string
1178
GUIBaseVehicle::getDeviceDescription() {
1179
std::vector<std::string> devs;
1180
for (MSDevice* d : myVehicle.getDevices()) {
1181
devs.push_back(d->deviceName());
1182
}
1183
return joinToString(devs, " ");
1184
}
1185
1186
1187
void
1188
GUIBaseVehicle::computeSeats(const Position& front, const Position& back, double seatOffset, int maxSeats, double exaggeration, int& requiredSeats, Seats& into, double extraOffset) const {
1189
if (requiredSeats <= 0) {
1190
return;
1191
}
1192
maxSeats = MAX2(maxSeats, 1); // compute at least one seat
1193
seatOffset *= exaggeration;
1194
const double vehWidth = getVType().getSeatingWidth() * exaggeration;
1195
const double length = front.distanceTo2D(back);
1196
const int rowSize = MAX2(1, (int)floor(vehWidth / seatOffset));
1197
const double frontSeatPos = getVType().getFrontSeatPos() + extraOffset;
1198
const double rowOffset = MAX2(1.0, (length - frontSeatPos - 1)) / ceil((double)maxSeats / rowSize);
1199
const double sideOffset = (rowSize - 1) / 2.0 * seatOffset;
1200
double rowPos = frontSeatPos - rowOffset;
1201
double angle = back.angleTo2D(front);
1202
const int fillDirection = MSGlobals::gLefthand ? -1 : 1;
1203
//if (myVehicle.getID() == "v0") std::cout << SIMTIME << " seatOffset=" << seatOffset << " max=" << maxSeats << " ex=" << exaggeration << " req=" << requiredSeats << " rowSize=" << rowSize << " sideOffset=" << sideOffset << " front=" << front << " back=" << back << " a=" << angle << " da=" << RAD2DEG(angle) << "\n";
1204
for (int i = 0; requiredSeats > 0 && i < maxSeats; i++) {
1205
int seat = (i % rowSize);
1206
if (seat == 0) {
1207
rowPos += rowOffset;
1208
}
1209
into.push_back(Seat(PositionVector::positionAtOffset2D(front, back, rowPos, (sideOffset - seat * seatOffset) * fillDirection), angle));
1210
requiredSeats--;
1211
}
1212
}
1213
1214
1215
/****************************************************************************/
1216
1217