Eclipse SUMO - Simulation of Urban MObility
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
GNECalibratorDialog.cpp
Go to the documentation of this file.
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/****************************************************************************/
18// Dialog for edit calibrators
19/****************************************************************************/
20
25#include <netedit/GNENet.h>
26#include <netedit/GNEViewNet.h>
27#include <netedit/GNEUndoList.h>
32
33#include "GNECalibratorDialog.h"
35#include "GNERouteDialog.h"
37
38// ===========================================================================
39// FOX callback mapping
40// ===========================================================================
41
50
51// Object implementation
52FXIMPLEMENT(GNECalibratorDialog, GNEAdditionalDialog, GNECalibratorDialogMap, ARRAYNUMBER(GNECalibratorDialogMap))
53
54// ===========================================================================
55// member method definitions
56// ===========================================================================
57
59 GNEAdditionalDialog(editedCalibrator, false, 640, 480) {
60
61 // Create two columns, one for Routes and VehicleTypes, and other for Flows
62 FXHorizontalFrame* columns = new FXHorizontalFrame(myContentFrame, GUIDesignUniformHorizontalFrame);
63 FXVerticalFrame* columnLeft = new FXVerticalFrame(columns, GUIDesignAuxiliarFrame);
64 FXVerticalFrame* columnRight = new FXVerticalFrame(columns, GUIDesignAuxiliarFrame);
65
66 // create add button and label for routes
67 FXHorizontalFrame* buttonAndLabelRoute = new FXHorizontalFrame(columnLeft, GUIDesignAuxiliarHorizontalFrame);
69 new FXLabel(buttonAndLabelRoute, ("Add new " + toString(SUMO_TAG_ROUTE) + "s").c_str(), nullptr, GUIDesignLabelThick(JUSTIFY_NORMAL));
70
71 // Create table in left frame
72 myRouteList = new FXTable(columnLeft, this, MID_GNE_CALIBRATORDIALOG_TABLE_ROUTE, GUIDesignTableAdditionals);
73 myRouteList->setSelBackColor(FXRGBA(255, 255, 255, 255));
74 myRouteList->setSelTextColor(FXRGBA(0, 0, 0, 255));
75 myRouteList->setEditable(false);
76
77 // create add button and label for vehicle types
78 FXHorizontalFrame* buttonAndLabelVehicleType = new FXHorizontalFrame(columnLeft, GUIDesignAuxiliarHorizontalFrame);
80 new FXLabel(buttonAndLabelVehicleType, ("Add new " + toString(SUMO_TAG_VTYPE) + "s").c_str(), nullptr, GUIDesignLabelThick(JUSTIFY_NORMAL));
81
82 // Create table in left frame
83 myVehicleTypeList = new FXTable(columnLeft, this, MID_GNE_CALIBRATORDIALOG_TABLE_VEHICLETYPE, GUIDesignTableAdditionals);
84 myVehicleTypeList->setSelBackColor(FXRGBA(255, 255, 255, 255));
85 myVehicleTypeList->setSelTextColor(FXRGBA(0, 0, 0, 255));
86 myVehicleTypeList->setEditable(false);
87
88 // create add button and label for flows in right frame
89 FXHorizontalFrame* buttonAndLabelFlow = new FXHorizontalFrame(columnRight, GUIDesignAuxiliarHorizontalFrame);
91 myLabelFlow = new FXLabel(buttonAndLabelFlow, ("Add new " + toString(GNE_TAG_CALIBRATOR_FLOW) + "s").c_str(), nullptr, GUIDesignLabelThick(JUSTIFY_NORMAL));
92
93 // Create table in right frame
94 myFlowList = new FXTable(columnRight, this, MID_GNE_CALIBRATORDIALOG_TABLE_FLOW, GUIDesignTableAdditionals);
95 myFlowList->setSelBackColor(FXRGBA(255, 255, 255, 255));
96 myFlowList->setSelTextColor(FXRGBA(0, 0, 0, 255));
97 myFlowList->setEditable(false);
98
99 // update tables
100 updateRouteTable();
101 updateVehicleTypeTable();
102 updateFlowTable();
103
104 // start a undo list for editing local to this additional
105 initChanges();
106
107 // Open dialog as modal
108 openAsModalDialog();
109}
110
111
113
114
115long
116GNECalibratorDialog::onCmdAccept(FXObject*, FXSelector, void*) {
117 // accept changes before closing dialog
119 // Stop Modal
120 getApp()->stopModal(this, TRUE);
121 return 1;
122}
123
124
125long
126GNECalibratorDialog::onCmdCancel(FXObject*, FXSelector, void*) {
127 // cancel changes
129 // Stop Modal
130 getApp()->stopModal(this, FALSE);
131 return 1;
132}
133
134
135long
136GNECalibratorDialog::onCmdReset(FXObject*, FXSelector, void*) {
137 // reset changes
138 resetChanges();
139 // update tables
143 return 1;
144}
145
146
147long
148GNECalibratorDialog::onCmdAddRoute(FXObject*, FXSelector, void*) {
149 // create new calibrator route and configure it with modal GNERouteDialog
150 GNERouteDialog(new GNERoute(myEditedAdditional->getNet()), false); // NOSONAR, constructor returns after dialog has been closed
151 // update routes table
153 return 1;
154}
155
156
157long
158GNECalibratorDialog::onCmdClickedRoute(FXObject*, FXSelector, void*) {
159 // check if some delete button was pressed
160 for (int i = 0; i < (int)myEditedAdditional->getNet()->getViewNet()->getNet()->getAttributeCarriers()->getDemandElements().at(SUMO_TAG_ROUTE).size(); i++) {
161 // obtain rerouter
163 if (myRouteList->getItem(i, 2)->hasFocus()) {
164 // find all flows that contains route to delete as "route" parameter
165 std::vector<GNEAdditional*> calibratorFlowsToErase;
166 for (auto j : myEditedAdditional->getChildAdditionals()) {
167 if (j->getAttribute(SUMO_ATTR_ROUTE) == myRouteList->getItem(i, 0)->getText().text()) {
168 calibratorFlowsToErase.push_back(j);
169 }
170 }
171 // if there are flows that has route to remove as "route" parameter
172 if (calibratorFlowsToErase.size() > 0) {
173 // open question dialog box
174 const std::string msg = ("Deletion of " + toString(SUMO_TAG_ROUTE) + " '" + myRouteList->getItem(i, 0)->getText().text() + "' will remove " +
175 toString(calibratorFlowsToErase.size()) + " " + toString(GNE_TAG_CALIBRATOR_FLOW) + (calibratorFlowsToErase.size() > 1 ? ("s") : ("")) + ". Continue?");
176 FXuint answer = FXMessageBox::question(getApp(), MBOX_YES_NO, ("Remove " + toString(GNE_TAG_CALIBRATOR_FLOW) + "s").c_str(), "%s", msg.c_str());
177 if (answer != 1) { //1:yes, 2:no, 4:esc
178 // abort deletion of route
179 return 0;
180 } else {
181 // remove affected flows of calibrator flows
182 for (auto j : calibratorFlowsToErase) {
184 }
185 // remove route of calibrator routes
186 myEditedAdditional->getNet()->getViewNet()->getUndoList()->add(new GNEChange_DemandElement(routeToEdit, false), true);
187 // update flows and route table
190 return 1;
191 }
192 } else {
193 // remove route
194 myEditedAdditional->getNet()->getViewNet()->getUndoList()->add(new GNEChange_DemandElement(routeToEdit, false), true);
195 // update routes table
197 return 1;
198 }
199 } else if (myRouteList->getItem(i, 0)->hasFocus() || myRouteList->getItem(i, 1)->hasFocus()) {
200 // modify route of calibrator routes with modal dialog
201 GNERouteDialog(routeToEdit, true); // NOSONAR, constructor returns after dialog has been closed
202 // update routes table
204 // update Flows routes also because Route ID could be changed
206 return 1;
207 }
208 }
209 // nothing to do
210 return 0;
211}
212
213
214long
215GNECalibratorDialog::onCmdAddFlow(FXObject*, FXSelector, void*) {
216 // get routes and vTypes
219 // only add flow if there is at least a GNERoute (There is always a Vehicle Type)
220 if (routes.size() > 0) {
221 // create new calibrator and configure it with modal GNECalibratorFlowDialog
222 GNECalibratorFlowDialog(new GNECalibratorFlow(myEditedAdditional, defaultVType, routes.begin()->second), false); // NOSONAR, constructor returns after dialog has been closed
223 // update flows table
225 return 1;
226 } else {
227 throw ProcessError(TL("routes cannot be empty"));
228 }
229}
230
231
232long
233GNECalibratorDialog::onCmdClickedFlow(FXObject*, FXSelector, void*) {
234 // check if some delete button was pressed
235 for (int i = 0; i < (int)myEditedAdditional->getChildAdditionals().size(); i++) {
236 if (myFlowList->getItem(i, 2)->hasFocus()) {
237 // remove flow of calibrator flows
239 // update flows table
241 return 1;
242 } else if (myFlowList->getItem(i, 0)->hasFocus() || myFlowList->getItem(i, 1)->hasFocus()) {
243 // modify flow of calibrator flows (temporal) with modal dialog
244 GNECalibratorFlowDialog(myEditedAdditional->getChildAdditionals().at(i), true); // NOSONAR, constructor returns after dialog has been closed
245 // update flows table
247 return 1;
248 }
249 }
250 // nothing to do
251 return 0;
252}
253
254
255long
256GNECalibratorDialog::onCmdAddVehicleType(FXObject*, FXSelector, void*) {
257 // create a new Vehicle Type and configure it with modal GNEVehicleTypeDialog
259 GNEVehicleTypeDialog(new GNEVType(vehicleTypeID, myEditedAdditional->getNet(), SVC_PASSENGER), false); // NOSONAR, constructor returns after dialog has been closed
260 // update vehicle types table
262 return 1;
263}
264
265
266long
267GNECalibratorDialog::onCmdClickedVehicleType(FXObject*, FXSelector, void*) {
268 // check if some delete button was pressed
269 for (int i = 0; i < (int)myEditedAdditional->getNet()->getViewNet()->getNet()->getAttributeCarriers()->getDemandElements().at(SUMO_TAG_VTYPE).size(); i++) {
270 // obtain vehicle type
272 // Make sure that default vehicle isn't edited
273 if ((i == 0) && (myVehicleTypeList->getItem(i, 0)->hasFocus() || myVehicleTypeList->getItem(i, 1)->hasFocus() || myVehicleTypeList->getItem(i, 2)->hasFocus())) {
274 FXMessageBox::warning(getApp(), MBOX_OK,
275 ("Error editing default " + toString(SUMO_TAG_VTYPE)).c_str(), "%s",
276 ("Default " + toString(SUMO_TAG_VTYPE) + " cannot be either edited or deleted.").c_str());
277 } else if (myVehicleTypeList->getItem(i, 2)->hasFocus()) {
278 // find all flows that contains vehicle type to delete as "vehicle type" parameter
279 std::vector<GNEAdditional*> calibratorFlowsToErase;
280 for (auto j : myEditedAdditional->getChildAdditionals()) {
281 if (j->getAttribute(SUMO_ATTR_TYPE) == myVehicleTypeList->getItem(i, 0)->getText().text()) {
282 calibratorFlowsToErase.push_back(j);
283 }
284 }
285 // if there are flows that has vehicle type to remove as "vehicle type" parameter
286 if (calibratorFlowsToErase.size() > 0) {
287 const std::string msg = ("Deletion of " + toString(SUMO_TAG_VTYPE) + " '" + myVehicleTypeList->getItem(i, 0)->getText().text() + "' will remove " +
288 toString(calibratorFlowsToErase.size()) + " " + toString(GNE_TAG_CALIBRATOR_FLOW) + (calibratorFlowsToErase.size() > 1 ? ("s") : ("")) + ". Continue?");
289 FXuint answer = FXMessageBox::question(getApp(), MBOX_YES_NO, ("Remove " + toString(GNE_TAG_CALIBRATOR_FLOW) + "s").c_str(), "%s", msg.c_str());
290 if (answer != 1) { //1:yes, 2:no, 4:esc
291 // abort deletion of vehicle type
292 return 0;
293 } else {
294 // remove affected flows of calibrator flows
295 for (auto j : calibratorFlowsToErase) {
297 }
298 // remove vehicle type of calibrator vehicle types
300 // update flows and vehicle types table
303 return 1;
304 }
305 } else {
306 // remove vehicle type of calibrator vehicle types
308 // update vehicle types table
310 return 1;
311 }
312 } else if (myVehicleTypeList->getItem(i, 0)->hasFocus() || myVehicleTypeList->getItem(i, 1)->hasFocus()) {
313 // modify vehicle type with modal dialog
314 GNEVehicleTypeDialog(vType, true); // NOSONAR, constructor returns after dialog has been closed
315 // update vehicle types table
317 // update Flows routes also because VType ID could be changed
319 return 1;
320 }
321 }
322 // nothing to do
323 return 0;
324}
325
326
327void
329 // clear table
330 myRouteList->clearItems();
331 // set number of rows
333 // Configure list
334 myRouteList->setVisibleColumns(4);
335 myRouteList->setColumnWidth(0, 136);
336 myRouteList->setColumnWidth(1, 136);
337 myRouteList->setColumnWidth(2, GUIDesignHeight);
338 myRouteList->setColumnText(0, toString(SUMO_ATTR_ID).c_str());
339 myRouteList->setColumnText(1, toString(SUMO_ATTR_EDGES).c_str());
340 myRouteList->setColumnText(2, "");
341 myRouteList->getRowHeader()->setWidth(0);
342 // Declare index for rows and pointer to FXTableItem
343 int indexRow = 0;
344 FXTableItem* item = nullptr;
345 // iterate over routes
347 // Set ID
348 item = new FXTableItem(toString(route.second->getAttribute(SUMO_ATTR_ID)).c_str());
349 myRouteList->setItem(indexRow, 0, item);
350 // Set edges
351 item = new FXTableItem(toString(route.second->getAttribute(SUMO_ATTR_EDGES)).c_str());
352 myRouteList->setItem(indexRow, 1, item);
353 // set remove
354 item = new FXTableItem("", GUIIconSubSys::getIcon(GUIIcon::REMOVE));
355 item->setJustify(FXTableItem::CENTER_X | FXTableItem::CENTER_Y);
356 item->setEnabled(false);
357 myRouteList->setItem(indexRow, 2, item);
358 // Update index
359 indexRow++;
360 }
361 // enable or disable flow and label button
363}
364
365
366void
368 // clear table
369 myFlowList->clearItems();
370 // set number of rows
371 myFlowList->setTableSize(int(myEditedAdditional->getChildAdditionals().size()), 3);
372 // Configure list
373 myFlowList->setVisibleColumns(3);
374 myFlowList->setColumnWidth(0, 136);
375 myFlowList->setColumnWidth(1, 136);
376 myFlowList->setColumnWidth(2, GUIDesignHeight);
377 myFlowList->setColumnText(0, toString(SUMO_ATTR_TYPE).c_str());
378 myFlowList->setColumnText(1, toString(SUMO_ATTR_VCLASS).c_str());
379 myFlowList->setColumnText(2, "");
380 myFlowList->getRowHeader()->setWidth(0);
381 // Declare index for rows and pointer to FXTableItem
382 int indexRow = 0;
383 FXTableItem* item = nullptr;
384 // iterate over flows
385 for (auto i : myEditedAdditional->getChildAdditionals()) {
386 // Set vehicle type
387 item = new FXTableItem(i->getAttribute(SUMO_ATTR_TYPE).c_str());
388 myFlowList->setItem(indexRow, 0, item);
389 // Set route
390 item = new FXTableItem(i->getAttribute(SUMO_ATTR_ROUTE).c_str());
391 myFlowList->setItem(indexRow, 1, item);
392 // set remove
393 item = new FXTableItem("", GUIIconSubSys::getIcon(GUIIcon::REMOVE));
394 item->setJustify(FXTableItem::CENTER_X | FXTableItem::CENTER_Y);
395 item->setEnabled(false);
396 myFlowList->setItem(indexRow, 2, item);
397 // Update index
398 indexRow++;
399 }
400 // enable or disable flow and label button
402}
403
404
405void
407 // clear table
408 myVehicleTypeList->clearItems();
409 // set number of rows
411 // Configure list
412 myVehicleTypeList->setVisibleColumns(4);
413 myVehicleTypeList->setColumnWidth(0, 136);
414 myVehicleTypeList->setColumnWidth(1, 136);
415 myVehicleTypeList->setColumnWidth(2, GUIDesignHeight);
416 myVehicleTypeList->setColumnText(0, toString(SUMO_ATTR_ID).c_str());
417 myVehicleTypeList->setColumnText(1, toString(SUMO_ATTR_VCLASS).c_str());
418 myVehicleTypeList->setColumnText(2, "");
419 myVehicleTypeList->getRowHeader()->setWidth(0);
420 // Declare index for rows and pointer to FXTableItem
421 int indexRow = 0;
422 FXTableItem* item = nullptr;
423 // iterate over vehicle types
425 // Set id
426 item = new FXTableItem(vType.second->getAttribute(SUMO_ATTR_ID).c_str());
427 myVehicleTypeList->setItem(indexRow, 0, item);
428 // Set VClass
429 item = new FXTableItem(vType.second->getAttribute(SUMO_ATTR_VCLASS).c_str());
430 myVehicleTypeList->setItem(indexRow, 1, item);
431 // set remove icon except for default vehicle type
432 if (indexRow != 0) {
433 item = new FXTableItem("", GUIIconSubSys::getIcon(GUIIcon::REMOVE));
434 } else {
435 item = new FXTableItem("");
436 }
437 item->setJustify(FXTableItem::CENTER_X | FXTableItem::CENTER_Y);
438 item->setEnabled(false);
439 myVehicleTypeList->setItem(indexRow, 2, item);
440 // Update index
441 indexRow++;
442 }
443 // enable or disable flow and label button
445}
446
447
448void
450 // disable AddFlow button if no route is defined
452 myAddFlow->disable();
453 myFlowList->disable();
454 myLabelFlow->setText(TL("No routes defined"));
455 } else {
456 myAddFlow->enable();
457 myFlowList->enable();
458 myLabelFlow->setText(("Add new " + toString(GNE_TAG_CALIBRATOR_FLOW) + "s").c_str());
459 }
460}
461
462
463/****************************************************************************/
FXDEFMAP(GNECalibratorDialog) GNECalibratorDialogMap[]
@ MID_GNE_CALIBRATORDIALOG_TABLE_FLOW
change table flow
@ MID_GNE_CALIBRATORDIALOG_ADD_ROUTE
add new route
@ MID_GNE_CALIBRATORDIALOG_ADD_FLOW
@ MID_GNE_CALIBRATORDIALOG_ADD_VEHICLETYPE
add vehicle type
@ MID_GNE_CALIBRATORDIALOG_TABLE_ROUTE
change table route
@ MID_GNE_CALIBRATORDIALOG_TABLE_VEHICLETYPE
change table route
#define GUIDesignButtonIcon
button only with icon
Definition GUIDesigns.h:91
#define GUIDesignTableAdditionals
design for tables used in additional dialogs
Definition GUIDesigns.h:634
#define GUIDesignAuxiliarHorizontalFrame
design for auxiliar (Without borders) horizontal frame used to pack another frames
Definition GUIDesigns.h:399
#define GUIDesignUniformHorizontalFrame
design for horizontal frame used to pack another frames with a uniform width
Definition GUIDesigns.h:411
#define GUIDesignLabelThick(justify)
label extended over frame with thick and with text justify to left
Definition GUIDesigns.h:249
#define GUIDesignAuxiliarFrame
design for auxiliar (Without borders) frame extended in all directions
Definition GUIDesigns.h:390
#define TL(string)
Definition MsgHandler.h:301
@ SVC_PASSENGER
vehicle is a passenger car (a "normal" car)
@ SUMO_TAG_VTYPE
description of a vehicle/person/container type
@ SUMO_TAG_ROUTE
begin/end of the description of a route
@ GNE_TAG_CALIBRATOR_FLOW
a flow definition within in Calibrator
@ SUMO_ATTR_EDGES
the edges of a route
@ SUMO_ATTR_VCLASS
@ SUMO_ATTR_TYPE
@ SUMO_ATTR_ROUTE
@ SUMO_ATTR_ID
int GUIDesignHeight
the default size for GUI elements
Definition StdDefs.cpp:35
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
Definition ToString.h:46
Dialog to edit sequences, parameters, etc.. of Additionals.
void acceptChanges()
Accept changes did in this dialog.
void cancelChanges()
Cancel changes did in this dialog.
GNEAdditional * myEditedAdditional
pointer to edited additional
void resetChanges()
reset changes did in this dialog.
GNENet * getNet() const
get pointer to net
Dialog for edit calibrators.
long onCmdReset(FXObject *, FXSelector, void *)
event after press reset button
long onCmdCancel(FXObject *, FXSelector, void *)
event after press cancel button
FXTable * myFlowList
list with flows
long onCmdClickedFlow(FXObject *, FXSelector, void *)
remove or edit flow
FXLabel * myLabelFlow
label for flows
long onCmdClickedVehicleType(FXObject *, FXSelector, void *)
remove or edit vehicle type
FXTable * myVehicleTypeList
list with vehicle types
long onCmdAccept(FXObject *, FXSelector, void *)
long onCmdAddVehicleType(FXObject *, FXSelector, void *)
add new vehicle type
long onCmdClickedRoute(FXObject *, FXSelector, void *)
remove or edit route
void updateRouteTable()
update data table with routes
void updateFlowAndLabelButton()
update flow and label button
FXTable * myRouteList
list with routes
long onCmdAddFlow(FXObject *, FXSelector, void *)
add new flow
long onCmdAddRoute(FXObject *, FXSelector, void *)
add new route
FXButton * myAddFlow
button for add new flow
void updateFlowTable()
update data table with flows
void updateVehicleTypeTable()
update data table with vehicle types
Dialog for editing calibrator flows.
const GNEHierarchicalContainerChildren< GNEAdditional * > & getChildAdditionals() const
return child additionals
const std::unordered_map< SumoXMLTag, std::unordered_map< const GUIGlObject *, GNEDemandElement * >, std::hash< int > > & getDemandElements() const
get demand elements
std::string generateDemandElementID(SumoXMLTag tag) const
generate demand element id
GNEDemandElement * retrieveDemandElement(SumoXMLTag type, const std::string &id, bool hardFail=true) const
Returns the named demand element.
GNEDemandElement * getDefaultType() const
get default type
GNENetHelper::AttributeCarriers * getAttributeCarriers() const
get all attribute carriers used in this net
Definition GNENet.cpp:146
GNEViewNet * getViewNet() const
get view net
Definition GNENet.cpp:2184
Dialog for editing Calibrator Routes.
void add(GNEChange *command, bool doit=false, bool merge=true)
Add new command, executing it if desired. The new command will be merged with the previous command if...
Dialog for editing calibrator vehicle types.
GNENet * getNet() const
get the net object
GNEUndoList * getUndoList() const
get the undoList object
static FXButton * buildFXButton(FXComposite *p, const std::string &text, const std::string &tip, const std::string &help, FXIcon *ic, FXObject *tgt, FXSelector sel, FXuint opts=BUTTON_NORMAL, FXint x=0, FXint y=0, FXint w=0, FXint h=0, FXint pl=DEFAULT_PAD, FXint pr=DEFAULT_PAD, FXint pt=DEFAULT_PAD, FXint pb=DEFAULT_PAD)
build button
static FXIcon * getIcon(const GUIIcon which)
returns a icon previously defined in the enum GUIIcon