diff --git a/src/de/schildbach/pte/AbstractEfaProvider.java b/src/de/schildbach/pte/AbstractEfaProvider.java
index 8ff40be8..25ef6e68 100644
--- a/src/de/schildbach/pte/AbstractEfaProvider.java
+++ b/src/de/schildbach/pte/AbstractEfaProvider.java
@@ -84,11 +84,18 @@ public abstract class AbstractEfaProvider implements NetworkProvider
// parse odv name elements
if (!XmlPullUtil.jumpToStartTag(pp, null, "itdOdv") || !"origin".equals(pp.getAttributeValue(null, "usage")))
throw new IllegalStateException("cannot find ");
- if (!XmlPullUtil.nextStartTagInsideTree(pp, null, "itdOdvName"))
+ XmlPullUtil.enter(pp, "itdOdv");
+ XmlPullUtil.enter(pp, "itdOdvPlace");
+ XmlPullUtil.exit(pp, "itdOdvPlace");
+ if (!XmlPullUtil.test(pp, "itdOdvName"))
throw new IllegalStateException("cannot find ");
- final String nameState = pp.getAttributeValue(null, "state");
+ final String nameState = XmlPullUtil.attr(pp, "state");
+ XmlPullUtil.enter(pp, "itdOdvName");
+ if (XmlPullUtil.test(pp, "itdMessage"))
+ XmlPullUtil.next(pp);
+
if ("identified".equals(nameState) || "list".equals(nameState))
- while (XmlPullUtil.nextStartTagInsideTree(pp, null, "odvNameElem"))
+ while (XmlPullUtil.test(pp, "odvNameElem"))
results.add(processOdvNameElem(pp));
// parse assigned stops
@@ -119,6 +126,9 @@ public abstract class AbstractEfaProvider implements NetworkProvider
private Location processOdvNameElem(final XmlPullParser pp) throws XmlPullParserException, IOException
{
+ if (!XmlPullUtil.test(pp, "odvNameElem"))
+ throw new IllegalStateException("expecting ");
+
final String anyType = pp.getAttributeValue(null, "anyType");
final String idStr = pp.getAttributeValue(null, "id");
final String stopIdStr = pp.getAttributeValue(null, "stopID");
@@ -179,7 +189,9 @@ public abstract class AbstractEfaProvider implements NetworkProvider
throw new IllegalArgumentException("unknown type: " + anyType + " " + idStr + " " + stopIdStr);
}
- final String name = normalizeLocationName(pp.nextText());
+ XmlPullUtil.enter(pp, "odvNameElem");
+ final String name = normalizeLocationName(pp.getText());
+ XmlPullUtil.exit(pp, "odvNameElem");
return new Location(type, id, lat, lon, name);
}
@@ -543,6 +555,8 @@ public abstract class AbstractEfaProvider implements NetworkProvider
return 'R' + str;
if (type.equals("ÖBA")) // Eisenbahn-Betriebsgesellschaft Ochsenhausen
return 'R' + str;
+ if (type.equals("MBS")) // Montafonerbahn
+ return 'R' + str;
if (type.equals("Abellio-Zug")) // Abellio
return 'R' + str;
if (type.equals("KBS")) // Kursbuchstrecke
@@ -621,6 +635,9 @@ public abstract class AbstractEfaProvider implements NetworkProvider
if (type.equals("STR")) // Nordhausen
return 'T' + str;
+ if (type.equals("BUS"))
+ return 'B' + str;
+
if (type.length() == 0)
return "?";
if (P_LINE_NUMBER.matcher(type).matches())
@@ -891,79 +908,98 @@ public abstract class AbstractEfaProvider implements NetworkProvider
if (!XmlPullUtil.jumpToStartTag(pp, null, "itdTripRequest"))
throw new IllegalStateException("cannot find ");
+ XmlPullUtil.enter(pp, "itdTripRequest");
+
+ if (XmlPullUtil.test(pp, "itdMessage"))
+ {
+ final int code = XmlPullUtil.intAttr(pp, "code");
+ if (code == -4000) // no connection
+ return new QueryConnectionsResult(Status.NO_CONNECTIONS);
+ XmlPullUtil.next(pp);
+ }
+ if (XmlPullUtil.test(pp, "itdPrintConfiguration"))
+ XmlPullUtil.next(pp);
+ if (XmlPullUtil.test(pp, "itdAddress"))
+ XmlPullUtil.next(pp);
// parse odv name elements
List ambiguousFrom = null, ambiguousTo = null, ambiguousVia = null;
Location from = null, via = null, to = null;
- if (!XmlPullUtil.jumpToStartTag(pp, null, "itdOdv") || !"origin".equals(pp.getAttributeValue(null, "usage")))
- throw new IllegalStateException("cannot find ");
- XmlPullUtil.nextStartTagInsideTree(pp, null, "itdOdvName");
- final String originState = pp.getAttributeValue(null, "state");
- if ("list".equals(originState))
+ while (XmlPullUtil.test(pp, "itdOdv"))
{
- ambiguousFrom = new ArrayList();
- while (XmlPullUtil.nextStartTagInsideTree(pp, null, "odvNameElem"))
- ambiguousFrom.add(processOdvNameElem(pp));
- }
- else if ("identified".equals(originState))
- {
- if (!XmlPullUtil.nextStartTagInsideTree(pp, null, "odvNameElem"))
- throw new IllegalStateException("cannot find ");
- from = processOdvNameElem(pp);
- }
+ final String usage = XmlPullUtil.attr(pp, "usage");
+ XmlPullUtil.enter(pp, "itdOdv");
+ XmlPullUtil.enter(pp, "itdOdvPlace");
+ XmlPullUtil.exit(pp, "itdOdvPlace");
+ if (!XmlPullUtil.test(pp, "itdOdvName"))
+ throw new IllegalStateException("cannot find inside " + usage);
+ final String state = XmlPullUtil.attr(pp, "state");
+ XmlPullUtil.enter(pp, "itdOdvName");
+ if (XmlPullUtil.test(pp, "itdMessage"))
+ XmlPullUtil.next(pp);
- if (!XmlPullUtil.jumpToStartTag(pp, null, "itdOdv") || !"destination".equals(pp.getAttributeValue(null, "usage")))
- throw new IllegalStateException("cannot find ");
- XmlPullUtil.nextStartTagInsideTree(pp, null, "itdOdvName");
- final String destinationState = pp.getAttributeValue(null, "state");
- if ("list".equals(destinationState))
- {
- ambiguousTo = new ArrayList();
- while (XmlPullUtil.nextStartTagInsideTree(pp, null, "odvNameElem"))
- ambiguousTo.add(processOdvNameElem(pp));
- }
- else if ("identified".equals(destinationState))
- {
- if (!XmlPullUtil.nextStartTagInsideTree(pp, null, "odvNameElem"))
- throw new IllegalStateException("cannot find ");
- to = processOdvNameElem(pp);
- }
+ if ("list".equals(state))
+ {
+ if ("origin".equals(usage))
+ {
+ ambiguousFrom = new ArrayList();
+ while (XmlPullUtil.test(pp, "odvNameElem"))
+ ambiguousFrom.add(processOdvNameElem(pp));
+ }
+ else if ("via".equals(usage))
+ {
+ ambiguousVia = new ArrayList();
+ while (XmlPullUtil.test(pp, "odvNameElem"))
+ ambiguousVia.add(processOdvNameElem(pp));
+ }
+ else if ("destination".equals(usage))
+ {
+ ambiguousTo = new ArrayList();
+ while (XmlPullUtil.test(pp, "odvNameElem"))
+ ambiguousTo.add(processOdvNameElem(pp));
+ }
+ else
+ {
+ throw new IllegalStateException("unknown usage: " + usage);
+ }
+ }
+ else if ("identified".equals(state))
+ {
+ if (!XmlPullUtil.test(pp, "odvNameElem"))
+ throw new IllegalStateException("cannot find inside " + usage);
- if (!XmlPullUtil.jumpToStartTag(pp, null, "itdOdv") || !"via".equals(pp.getAttributeValue(null, "usage")))
- throw new IllegalStateException("cannot find ");
- XmlPullUtil.nextStartTagInsideTree(pp, null, "itdOdvName");
- final String viaState = pp.getAttributeValue(null, "state");
- if ("list".equals(viaState))
- {
- ambiguousVia = new ArrayList();
- while (XmlPullUtil.nextStartTagInsideTree(pp, null, "odvNameElem"))
- ambiguousVia.add(processOdvNameElem(pp));
- }
- else if ("identified".equals(viaState))
- {
- if (!XmlPullUtil.nextStartTagInsideTree(pp, null, "odvNameElem"))
- throw new IllegalStateException("cannot find ");
- via = processOdvNameElem(pp);
+ if ("origin".equals(usage))
+ from = processOdvNameElem(pp);
+ else if ("via".equals(usage))
+ via = processOdvNameElem(pp);
+ else if ("destination".equals(usage))
+ to = processOdvNameElem(pp);
+ else
+ throw new IllegalStateException("unknown usage: " + usage);
+ }
+ XmlPullUtil.exit(pp, "itdOdvName");
+ XmlPullUtil.exit(pp, "itdOdv");
}
if (ambiguousFrom != null || ambiguousTo != null || ambiguousVia != null)
return new QueryConnectionsResult(ambiguousFrom, ambiguousVia, ambiguousTo);
- if (!XmlPullUtil.jumpToStartTag(pp, null, "itdTripDateTime"))
- throw new IllegalStateException("cannot find ");
- if (!XmlPullUtil.nextStartTagInsideTree(pp, null, "itdDateTime"))
- throw new IllegalStateException("cannot find ");
- if (!XmlPullUtil.nextStartTagInsideTree(pp, null, "itdDate"))
+ XmlPullUtil.enter(pp, "itdTripDateTime");
+ XmlPullUtil.enter(pp, "itdDateTime");
+ if (!XmlPullUtil.test(pp, "itdDate"))
throw new IllegalStateException("cannot find ");
if (!pp.isEmptyElementTag())
{
- if (!XmlPullUtil.nextStartTagInsideTree(pp, null, "itdMessage"))
+ XmlPullUtil.enter(pp, "itdDate");
+ if (!XmlPullUtil.test(pp, "itdMessage"))
throw new IllegalStateException("cannot find ");
final String message = pp.nextText();
if (message.equals("invalid date"))
return new QueryConnectionsResult(Status.INVALID_DATE);
+ XmlPullUtil.exit(pp, "itdDate");
}
+ XmlPullUtil.exit(pp, "itdDateTime");
final Calendar departureTime = new GregorianCalendar(), arrivalTime = new GregorianCalendar(), stopTime = new GregorianCalendar();
departureTime.setTimeZone(timeZone());
@@ -1174,6 +1210,7 @@ public abstract class AbstractEfaProvider implements NetworkProvider
protected static final void appendCommonConnectionParams(final StringBuilder uri)
{
uri.append("&outputFormat=XML");
+ uri.append("&coordListOutputFormat=STRING");
uri.append("&coordOutputFormat=WGS84");
uri.append("&calcNumberOfTrips=4");
}
diff --git a/src/de/schildbach/pte/VorProvider.java b/src/de/schildbach/pte/VorProvider.java
new file mode 100644
index 00000000..3d5bb824
--- /dev/null
+++ b/src/de/schildbach/pte/VorProvider.java
@@ -0,0 +1,126 @@
+/*
+ * Copyright 2010 the original author or authors.
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ */
+package de.schildbach.pte;
+
+import java.text.DateFormat;
+import java.text.SimpleDateFormat;
+import java.util.Date;
+
+import de.schildbach.pte.dto.Location;
+import de.schildbach.pte.util.ParserUtils;
+
+/**
+ * @author Andreas Schildbach
+ */
+public class VorProvider extends AbstractEfaProvider
+{
+ public static final String NETWORK_ID = "efa.vor.at";
+ private final static String API_BASE = "http://efa.vor.at/wvb/";
+
+ public boolean hasCapabilities(Capability... capabilities)
+ {
+ for (final Capability capability : capabilities)
+ if (capability == Capability.DEPARTURES || capability == Capability.CONNECTIONS)
+ return true;
+
+ return false;
+ }
+
+ private static final String AUTOCOMPLETE_URI = API_BASE
+ + "XSLT_TRIP_REQUEST2?outputFormat=XML&coordOutputFormat=WGS84&locationServerActive=1&type_origin=any&name_origin=%s";
+
+ @Override
+ protected String autocompleteUri(final CharSequence constraint)
+ {
+ return String.format(AUTOCOMPLETE_URI, ParserUtils.urlEncode(constraint.toString(), "ISO-8859-1"));
+ }
+
+ @Override
+ protected String nearbyLatLonUri(final int lat, final int lon)
+ {
+ return null;
+ }
+
+ private static final String NEARBY_STATION_URI = API_BASE
+ + "XSLT_DM_REQUEST"
+ + "?outputFormat=XML&coordOutputFormat=WGS84&name_dm=%s&type_dm=stop&itOptionsActive=1&ptOptionsActive=1&useProxFootSearch=1&mergeDep=1&useAllStops=1&mode=direct";
+
+ @Override
+ protected String nearbyStationUri(final String stationId)
+ {
+ return String.format(NEARBY_STATION_URI, ParserUtils.urlEncode(stationId));
+ }
+
+ @Override
+ protected String departuresQueryUri(final String stationId, final int maxDepartures)
+ {
+ final StringBuilder uri = new StringBuilder();
+ uri.append(API_BASE).append("XSLT_DM_REQUEST");
+ uri.append("?outputFormat=XML");
+ uri.append("&coordOutputFormat=WGS84");
+ uri.append("&type_dm=stop");
+ uri.append("&name_dm=").append(stationId);
+ uri.append("&mode=direct");
+ uri.append("&useRealtime=1");
+ return uri.toString();
+ }
+
+ @Override
+ protected String connectionsQueryUri(final Location from, final Location via, final Location to, final Date date, final boolean dep,
+ final String products, final WalkSpeed walkSpeed)
+ {
+ final DateFormat DATE_FORMAT = new SimpleDateFormat("yyyyMMdd");
+ final DateFormat TIME_FORMAT = new SimpleDateFormat("HHmm");
+
+ final StringBuilder uri = new StringBuilder();
+ uri.append(API_BASE);
+ uri.append("XSLT_TRIP_REQUEST2");
+
+ uri.append("?language=de");
+ appendCommonConnectionParams(uri);
+
+ appendLocation(uri, from, "origin");
+ appendLocation(uri, to, "destination");
+ if (via != null)
+ appendLocation(uri, via, "via");
+
+ uri.append("&itdDate=").append(ParserUtils.urlEncode(DATE_FORMAT.format(date)));
+ uri.append("&itdTime=").append(ParserUtils.urlEncode(TIME_FORMAT.format(date)));
+ uri.append("&itdTripDateTimeDepArr=").append(dep ? "dep" : "arr");
+
+ uri.append("&ptOptionsActive=1");
+ uri.append("&changeSpeed=").append(WALKSPEED_MAP.get(walkSpeed));
+ uri.append(productParams(products));
+
+ uri.append("&locationServerActive=1");
+ uri.append("&useRealtime=1");
+
+ return uri.toString();
+ }
+
+ @Override
+ protected String commandLink(final String sessionId, final String command)
+ {
+ final StringBuilder uri = new StringBuilder();
+ uri.append(API_BASE);
+ uri.append("XSLT_TRIP_REQUEST2");
+ uri.append("?sessionID=").append(sessionId);
+ appendCommonConnectionParams(uri);
+ uri.append("&command=").append(command);
+ return uri.toString();
+ }
+}