summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGravatar Dirk Hohndel <dirk@hohndel.org>2013-02-04 08:31:45 +1100
committerGravatar Dirk Hohndel <dirk@hohndel.org>2013-02-04 08:31:45 +1100
commitdb05dd78f3cadaea32e260f68ad50e1c0da21526 (patch)
tree79ad02d839358b6581e0a3bdaaee2ffb69343007
parent3a3880cf256ec669b9d597a37e54bb106dd928f8 (diff)
downloadsubsurface-db05dd78f3cadaea32e260f68ad50e1c0da21526.tar.gz
Better fix for translated cardinal directions
The previous attempt in commit f305c5c83fe1 ("Correctly parse translated cardinal directions") suffered from a bit of false advertising in that it wasn't, actually, "correct". It made silly assumptions about the length of the translated strings being 1 and also forgot the middle part of the algorithm where we use the appearance of 'E' or 'W' (and their translated brethren) as indication that there are no minutes for the the latitude. Hopefully this version does better. Reported-by: Sergey Starosek <sergey.starosek@gmail.com> Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
-rw-r--r--info.c40
1 files changed, 27 insertions, 13 deletions
diff --git a/info.c b/info.c
index c212316fe..bc8ae5440 100644
--- a/info.c
+++ b/info.c
@@ -380,6 +380,24 @@ static int get_rating(const char *string)
return rating_val;
}
+/* this is used to skip the cardinal directions (or check if they are
+ * present). You pass in the text and a STRING with the direction.
+ * This checks for both the standard english text (just one character)
+ * and the translated text (possibly longer) and returns 0 if not found
+ * and the number of chars to skip otherwise. */
+static int string_advance_cardinal(const char *text, const char *look)
+{
+ char *trans;
+ int len = strlen(look);
+ if (!strncasecmp(text, look, len))
+ return len;
+ trans = _(look);
+ len = strlen(trans);
+ if (!strncasecmp(text, trans, len))
+ return len;
+ return 0;
+}
+
/* this has to be done with UTF8 as people might want to enter the degree symbol */
static gboolean parse_gps_text(const char *gps_text, double *latitude, double *longitude)
{
@@ -390,6 +408,7 @@ static gboolean parse_gps_text(const char *gps_text, double *latitude, double *l
double parselat, parselong;
gunichar degrees = UCS4_DEGREE;
gunichar c;
+ int incr;
while (g_unichar_isspace(g_utf8_get_char(text)))
text = g_utf8_next_char(text);
@@ -401,12 +420,9 @@ static gboolean parse_gps_text(const char *gps_text, double *latitude, double *l
return TRUE;
}
/* ok, let's parse by hand - first degrees of latitude */
- if (g_unichar_toupper(g_utf8_get_char(text)) == 'N' ||
- !strncmp(text, _("N"), strlen(_("N"))))
- text++;
- if (g_unichar_toupper(g_utf8_get_char(text)) == 'S' ||
- !strncmp(text, _("S"), strlen(_("S")))) {
- text++;
+ text += string_advance_cardinal(text, "N");
+ if ((incr = string_advance_cardinal(text, "S")) > 0) {
+ text += incr;
south = TRUE;
}
parselat = strtod(text, &endptr);
@@ -421,7 +437,8 @@ static gboolean parse_gps_text(const char *gps_text, double *latitude, double *l
/* next optional minutes as decimal, skipping degree symbol */
while (g_unichar_isspace(c = g_utf8_get_char(text)) || c == degrees)
text = g_utf8_next_char(text);
- if (g_unichar_toupper(c) != 'E' && g_unichar_toupper(c) != 'W' && c != ';' && c != ',') {
+ incr = string_advance_cardinal(text, "E") + string_advance_cardinal(text, "W");
+ if (!incr && c != ';' && c != ',') {
parselat += strtod(text, &endptr) / 60.0;
if (text == endptr)
return FALSE;
@@ -435,12 +452,9 @@ static gboolean parse_gps_text(const char *gps_text, double *latitude, double *l
text = g_utf8_next_char(text);
/* next degrees of longitude */
- if (g_unichar_toupper(g_utf8_get_char(text)) == 'E' ||
- !strncmp(text, _("E"), strlen(_("E"))))
- text++;
- if (g_unichar_toupper(g_utf8_get_char(text)) == 'W' ||
- !strncmp(text, _("W"), strlen(_("W")))) {
- text++;
+ text += string_advance_cardinal(text, "E");
+ if ((incr = string_advance_cardinal(text, "W")) > 0) {
+ text += incr;
west = TRUE;
}
parselong = strtod(text, &endptr);