#include <config.h>
#include <string>
#include <utils/options/Option.h>
#include <utils/options/OptionsCont.h>
#include <utils/common/MsgHandler.h>
#include <utils/common/SystemFrame.h>
#include <utils/common/SysUtils.h>
#include <utils/iodevices/OutputDevice.h>
#include <netbuild/NBNetBuilder.h>
#include "NWFrame.h"
#include "NWWriter_SUMO.h"
#include "NWWriter_Amitran.h"
#include "NWWriter_MATSim.h"
#include "NWWriter_XML.h"
#include "NWWriter_OpenDrive.h"
#include "NWWriter_DlrNavteq.h"
void
NWFrame::fillOptions(OptionsCont& oc, bool forNetgen) {
oc.doRegister("output-file", 'o', new Option_FileName());
oc.addSynonyme("output-file", "sumo-output");
oc.addSynonyme("output-file", "output");
oc.addDescription("output-file", "Output", TL("The generated net will be written to FILE"));
oc.doRegister("plain-output-prefix", 'p', new Option_FileName());
oc.addSynonyme("plain-output-prefix", "plain-output");
oc.addSynonyme("plain-output-prefix", "plain");
oc.addDescription("plain-output-prefix", "Output", TL("Prefix of files to write plain xml nodes, edges and connections to"));
oc.doRegister("plain-output.lanes", new Option_Bool(false));
oc.addDescription("plain-output.lanes", "Output", TL("Write all lanes and their attributes even when they are not customized"));
oc.doRegister("junctions.join-output", new Option_FileName());
oc.addDescription("junctions.join-output", "Output",
"Writes information about joined junctions to FILE (can be loaded as additional node-file to reproduce joins");
oc.doRegister("prefix", new Option_String(""));
oc.addDescription("prefix", "Output", TL("Defines a prefix for edge and junction IDs"));
oc.doRegister("prefix.junction", new Option_String(""));
oc.addDescription("prefix.junction", "Output", TL("Defines a prefix for junction IDs"));
oc.doRegister("prefix.edge", new Option_String(""));
oc.addDescription("prefix.edge", "Output", TL("Defines a prefix for edge IDs"));
#ifdef PROJ_API_FILE
if (!forNetgen) {
oc.doRegister("proj.plain-geo", new Option_Bool(false));
oc.addDescription("proj.plain-geo", "Projection", TL("Write geo coordinates in plain-xml"));
}
#endif
oc.doRegister("amitran-output", new Option_FileName());
oc.addDescription("amitran-output", "Output", TL("The generated net will be written to FILE using Amitran format"));
oc.doRegister("matsim-output", new Option_FileName());
oc.addDescription("matsim-output", "Output", TL("The generated net will be written to FILE using MATSim format"));
oc.doRegister("opendrive-output", new Option_FileName());
oc.addDescription("opendrive-output", "Output", TL("The generated net will be written to FILE using OpenDRIVE format"));
oc.doRegister("dlr-navteq-output", new Option_FileName());
oc.addDescription("dlr-navteq-output", "Output", TL("The generated net will be written to dlr-navteq files with the given PREFIX"));
oc.doRegister("dlr-navteq.version", new Option_String("6.5"));
oc.addDescription("dlr-navteq.version", "Output", TL("The dlr-navteq output format version to write"));
oc.doRegister("dlr-navteq.precision", new Option_Integer(2));
oc.addDescription("dlr-navteq.precision", "Output", TL("The network coordinates are written with the specified level of output precision"));
oc.doRegister("output.street-names", new Option_Bool(false));
oc.addDescription("output.street-names", "Output", TL("Street names will be included in the output (if available)"));
oc.doRegister("output.original-names", new Option_Bool(false));
oc.addDescription("output.original-names", "Output", TL("Writes original names, if given, as parameter"));
oc.doRegister("output.removed-nodes", new Option_Bool(false));
oc.addDescription("output.removed-nodes", "Output", TL("Writes IDs of nodes remove with --geometry.remove into edge param"));
oc.doRegister("street-sign-output", new Option_FileName());
oc.addDescription("street-sign-output", "Output", TL("Writes street signs as POIs to FILE"));
if (!forNetgen) {
oc.doRegister("ptstop-output", new Option_FileName());
oc.addDescription("ptstop-output", "Output", TL("Writes public transport stops to FILE"));
oc.doRegister("ptline-output", new Option_FileName());
oc.addDescription("ptline-output", "Output", TL("Writes public transport lines to FILE"));
oc.doRegister("ptline-clean-up", new Option_Bool(false));
oc.addDescription("ptline-clean-up", "Output", TL("Clean-up pt stops that are not served by any line"));
oc.doRegister("parking-output", new Option_FileName());
oc.addDescription("parking-output", "Output", TL("Writes parking areas to FILE"));
oc.doRegister("railway.topology.output", new Option_FileName());
oc.addDescription("railway.topology.output", "Output", TL("Analyze topology of the railway network"));
oc.doRegister("polygon-output", new Option_FileName());
oc.addSynonyme("polygon-output", "taz-output");
oc.addDescription("polygon-output", "Output", TL("Write shapes that are embedded in the network input and that are not supported by polyconvert (OpenDRIVE)"));
}
oc.doRegister("opendrive-output.straight-threshold", new Option_Float(0.00000001));
oc.addDescription("opendrive-output.straight-threshold", "Output", TL("Builds parameterized curves whenever the angular change between straight segments exceeds FLOAT degrees"));
if (!forNetgen) {
oc.doRegister("opendrive-output.lefthand-left", new Option_Bool(false));
oc.addDescription("opendrive-output.lefthand-left", "Output", TL("Write lanes in lefthand networks on the left side (positive indices)"));
oc.doRegister("opendrive-output.shape-match-dist", new Option_Float(-1));
oc.addDescription("opendrive-output.shape-match-dist", "Output", TL("Match loaded shapes to the closest edge within FLOAT and export as road objects"));
}
}
bool
NWFrame::checkOptions(OptionsCont& oc) {
bool ok = true;
if (!oc.isSet("output-file")
&& !oc.isSet("plain-output-prefix")
&& !oc.isSet("amitran-output")
&& !oc.isSet("matsim-output")
&& !oc.isSet("opendrive-output")
&& !oc.isSet("dlr-navteq-output")) {
std::string net = "net.net.xml";
if (oc.isSet("configuration-file")) {
net = FileHelpers::getConfigurationRelative(oc.getString("configuration-file"), net);
}
oc.setDefault("output-file", net);
}
if (oc.isSet("opendrive-output") && oc.getBool("no-internal-links")) {
WRITE_ERROR(TL("OpenDRIVE export needs internal links computation."));
ok = false;
}
if (oc.isSet("opendrive-output") && oc.isDefault("no-internal-links")) {
oc.setDefault("no-internal-links", "false");
}
if (oc.isSet("opendrive-output") && oc.isDefault("rectangular-lane-cut")) {
oc.setDefault("rectangular-lane-cut", "true");
}
if (oc.isSet("opendrive-output") && !oc.getBool("rectangular-lane-cut")) {
WRITE_WARNING(TL("OpenDRIVE cannot represent oblique lane cuts and should use option 'rectangular-lane-cut'."));
}
if (oc.isSet("dlr-navteq-output") && oc.isDefault("numerical-ids")) {
oc.setDefault("numerical-ids", "true");
}
if (oc.isSet("dlr-navteq-output") && oc.isDefault("osm.all-attributes")) {
oc.setDefault("osm.all-attributes", "true");
oc.setDefault("osm.extra-attributes", "bridge,tunnel,layer,postal_code,maxheight,maxwidth,maxweight,surface");
}
if (oc.exists("ptline-output") && oc.isSet("ptline-output") && !oc.isSet("ptstop-output")) {
WRITE_ERROR(TL("public transport lines output requires 'ptstop-output' to be set"));
ok = false;
}
if (oc.exists("ptline-clean-up") && oc.getBool("ptline-clean-up") && !oc.isSet("ptline-output")) {
WRITE_WARNING(TL("'ptline-clean-up' only works in conjunction with 'ptline-output'. Ignoring invalid option."));
}
if (oc.isDefault("prefix.junction")) {
oc.setDefault("prefix.junction", oc.getString("prefix"));
}
if (oc.isDefault("prefix.edge")) {
oc.setDefault("prefix.edge", oc.getString("prefix"));
}
return ok;
}
void
NWFrame::writeNetwork(const OptionsCont& oc, NBNetBuilder& nb) {
const long before = PROGRESS_BEGIN_TIME_MESSAGE(TL("Writing network"));
NWWriter_SUMO::writeNetwork(oc, nb);
NWWriter_Amitran::writeNetwork(oc, nb);
NWWriter_MATSim::writeNetwork(oc, nb);
NWWriter_OpenDrive::writeNetwork(oc, nb);
NWWriter_DlrNavteq::writeNetwork(oc, nb);
NWWriter_XML::writeNetwork(oc, oc.isSet("plain-output-prefix") ? oc.getString("plain-output-prefix") : "", nb);
PROGRESS_TIME_MESSAGE(before);
}
void
NWFrame::writePositionLong(const Position& pos, OutputDevice& dev) {
dev.writeAttr(SUMO_ATTR_X, pos.x());
dev.writeAttr(SUMO_ATTR_Y, pos.y());
if (pos.z() != 0) {
dev.writeAttr(SUMO_ATTR_Z, pos.z());
}
}