]> err.no Git - mapper/blobdiff - src/poi-gui.c
Fixes to gstreamer element and caps handlings.
[mapper] / src / poi-gui.c
index 944f37042a3b617a616a22333f0817c15d09bf9c..3538295faa7428ffd118aabbb29386e3d8e2719b 100644 (file)
@@ -1,7 +1,5 @@
 #include <config.h>
 
-#define _GNU_SOURCE
-
 #include <unistd.h>
 #include <stdlib.h>
 #include <string.h>
@@ -17,6 +15,7 @@
 #include <libintl.h>
 #include <locale.h>
 #include <gtkhtml/gtkhtml.h>
+#include <gtkhtml/gtkhtml-stream.h>
 
 #include "utils.h"
 #include "poi.h"
 #include "settings.h"
 #include "poi-gui.h"
 #include "osm.h"
-
+#include "osm-db.h"
+#include "dialogs.h"
 #include "help.h"
+#include "track.h"
+#include "poi-gui.h"
 
 #ifdef WITH_OSSO
 #include <libosso.h>
@@ -41,8 +43,6 @@
 #include "ui-maemo.h"
 #endif
 
-#define POI_QUICK_BUTTONS (9)
-
 typedef struct _search_dialog search_dialog;
 struct _search_dialog {
        GtkWidget *dialog;
@@ -68,28 +68,28 @@ struct _quick_poi_categories {
     const gchar *name;
 };
 
+#define QPBS_X (4)
+#define QPBS_Y (3)
+#define POI_QUICK_BUTTONS (QPBS_X*QPBS_Y)
+
 static struct _quick_poi_categories quick_poi_categories[] = {
        { NODE_AMENITY_SPEEDCAM,        "Speedcam", },
        { NODE_AMENITY_FUEL,            "Fuel", },
        { NODE_AMENITY_PARKING,         "Parking", },
+       { NODE_AMENITY_TAXI,            "Taxi", },
 
        { NODE_AMENITY_PUB,             "Pub", },
        { NODE_AMENITY_CAFE,            "Cafe", },
-       { NODE_AMENITY_FOOD,            "Food", },
+       { NODE_AMENITY_FOOD,            "Fast Food", },
+       { NODE_AMENITY_RESTAURANT,      "Restaurant", },
 
        { NODE_AMENITY_SHOP,            "Shop", },
        { NODE_AMENITY_BANK,            "Bank", },
        { NODE_AMENITY_ATM,             "ATM", },
+       { NODE_AMENITY_POST_BOX,        "Post box", },
 };
 
-/* Quick POI information structure */
-typedef struct _poi_quick_data poi_quick_data;
-static struct _poi_quick_data {
-       GtkWidget *dialog;
-       GtkWidget *label;
-       gdouble lat;
-       gdouble lon;
-} qp;
+static poi_quick_data qp;
 
 static gboolean 
 category_delete(GtkWidget *widget, delete_poi *dpoi)
@@ -109,6 +109,7 @@ if (i == GTK_RESPONSE_OK) {
        if (poi_category_delete(dpoi->id)==FALSE)
                popup_error(_window, _("Problem deleting category or POI"));
        gtk_widget_hide_all(dpoi->dialog);
+       map_poi_cache_clear();
        map_force_redraw();
 }
 
@@ -358,7 +359,7 @@ GtkListStore *store;
 
 dialog = gtk_dialog_new_with_buttons(_("POI Categories"),
                             GTK_WINDOW(_window),
-                            GTK_DIALOG_MODAL, GTK_STOCK_OK,
+                            GTK_DIALOG_MODAL, GTK_STOCK_CLOSE,
                             GTK_RESPONSE_ACCEPT, NULL);
 
 help_dialog_help_enable(GTK_DIALOG(dialog), HELP_ID_POICAT);
@@ -376,8 +377,9 @@ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), GTK_POLICY_AUTOMATIC, GT
 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), sw, TRUE, TRUE, 0);
 
 tree_view=gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
-/* Maemo-related? */
+#ifdef WITH_HILDON
 g_object_set(tree_view, "allow-checkbox-mode", FALSE, NULL);
+#endif
 gtk_container_add(GTK_CONTAINER(sw), tree_view);
 
 gtk_tree_selection_set_mode(gtk_tree_view_get_selection (GTK_TREE_VIEW(tree_view)), GTK_SELECTION_SINGLE);
@@ -421,9 +423,7 @@ g_signal_connect(G_OBJECT(btn_edit), "clicked", G_CALLBACK(category_edit_cb), tr
 g_signal_connect(G_OBJECT(btn_add), "clicked", G_CALLBACK(category_add_cb), tree_view);
 
 gtk_widget_show_all(dialog);
-while (GTK_RESPONSE_ACCEPT == gtk_dialog_run(GTK_DIALOG(dialog))) {
-       break;
-}
+gtk_dialog_run(GTK_DIALOG(dialog));
 gtk_widget_destroy(dialog);
 
 return TRUE;
@@ -447,6 +447,7 @@ if (i == GTK_RESPONSE_OK) {
                popup_error(_window, _("Problem deleting POI"));
        } else {
                gtk_widget_hide_all(dpoi->dialog);
+               map_poi_cache_clear();
                map_force_redraw();
        }
 }
@@ -467,12 +468,12 @@ if (gtk_combo_box_get_active_iter(GTK_COMBO_BOX(sd->cmb_category), &iter)==TRUE)
 else
        cid=-1;
 
-if ((strlen(gtk_entry_get_text(sd->search_entry))<2) && (cid==-1)) {
+if ((strlen(gtk_entry_get_text(GTK_ENTRY(sd->search_entry)))<2) && (cid==-1)) {
        popup_error(sd->dialog, _("Query string too short. Minimum 2 characters."));
        return TRUE;
 }
 
-s=g_strdup(gtk_entry_get_text(sd->search_entry));
+s=g_strdup(gtk_entry_get_text(GTK_ENTRY(sd->search_entry)));
 g_printf("Search: [%s]\n", s);
 
 #if 0
@@ -692,8 +693,11 @@ if (poi_category_list())
 return TRUE;
 }
 
+/**
+ * Url click callback
+ */
 static void
-poi_info_url_cb(GtkHTML *html, const gchar *url, gpointer data)
+poi_info_url_clicked_cb(GtkHTML *html, const gchar *url, gpointer data)
 {
 g_printf("URL: %s\n", url);
 #ifdef WITH_OSSO
@@ -701,10 +705,37 @@ osso_rpc_run_with_defaults(_osso, "osso_browser",
        OSSO_BROWSER_OPEN_NEW_WINDOW_REQ, NULL,
        DBUS_TYPE_STRING, url, DBUS_TYPE_BOOLEAN, FALSE, DBUS_TYPE_INVALID);
 #else
-
+g_debug("Not yet implemented for !OSSO\n");
 #endif
 }
 
+
+/**
+ * Callback to load requested url
+ */
+static void
+poi_info_url_requested_cb(GtkHTML *html, const char *url, GtkHTMLStream *stream)
+{
+g_printf("URL: %s\n", url);
+gtk_html_stream_close(stream, GTK_HTML_STREAM_ERROR);
+}
+
+static void
+poi_info_title_cb(GtkHTML *html, const gchar *title)
+{
+g_printf("Title: %s\n", title);
+}
+
+#define WRITE_HTML(html, args...) \
+       { gchar *ph; \
+       ph=g_markup_printf_escaped(html, ##args); \
+       gtk_html_write(GTK_HTML(info), s, ph, strlen(ph)); \
+       g_free(ph); }
+
+/**
+ * Display a nice POI information dialog
+ *
+ */
 gboolean
 poi_info_dialog(guint poi_id)
 {
@@ -712,43 +743,18 @@ GtkWidget *dialog;
 GtkWidget *info;
 GtkWidget *sw;
 poi_info *p;
-gchar *phtml;
+GtkHTMLStream *s;
 gint ls;
 
 p=poi_get_by_id(poi_id);
-if (!p)
+if (!p) {
+       g_debug("Requested info for POI %d that does not exist!\n", poi_id);
        return FALSE;
-
-ls=strlen(p->label);
-
-phtml=g_strdup_printf(
-       "<html><meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">"
-       "<body>"
-       "<b>%s%s</b><br><table>"
-       "<tr><th align=\"left\">Location:</th><td>%f, %f</td></tr>"
-       "<tr><th align=\"left\">Public:</th><td>%s</td></tr>"
-       "<tr><th align=\"left\">Source:</th><td>%s</td></tr>"
-       "<tr><th align=\"left\">Category:</th><td>%s</td></tr>"
-       "<tr><th align=\"left\">Description:</th><td>%s</td></tr>"
-       "<tr><th align=\"left\">URL:</th><td><a href=\"%s\">%s</a></td></tr>"
-       "<tr><th align=\"left\">Postal Code:</th><td>%s</td></tr>"
-       "</table></body></html>",
-       ls>0 ? p->label : p->cat_label,
-       ls==0 ? " (No name)" : "",
-       p->lat, p->lon,
-       p->public==1 ? "Yes" : "No",
-       p->source==POI_SOURCE_OSM ? "OpenStreetMap" : "Other",
-       p->cat_label ? p->cat_label : "",
-       p->desc ? p->desc : "",
-       p->url ? p->url : "",
-       p->url ? p->url : "",
-       p->postal_code ? p->postal_code : "");
+}
 
 dialog=gtk_dialog_new_with_buttons(_("POI"),
-                       GTK_WINDOW(_window),
-                       GTK_DIALOG_MODAL,
-                       GTK_STOCK_OK,
-                       GTK_RESPONSE_ACCEPT,
+                       GTK_WINDOW(_window), GTK_DIALOG_MODAL,
+                       GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
                        NULL);
 
 /* XXX: Add edit button */
@@ -761,20 +767,49 @@ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), GTK_POLICY_AUTOMATIC, GT
 gtk_container_add(GTK_CONTAINER(sw), info);
 gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), sw, TRUE, TRUE, 0);
 
-g_signal_connect(G_OBJECT(info), "link_clicked", G_CALLBACK(poi_info_url_cb), NULL);
+g_signal_connect(G_OBJECT(info), "link_clicked", G_CALLBACK(poi_info_url_clicked_cb), NULL);
+g_signal_connect(G_OBJECT(info), "url_requested", G_CALLBACK(poi_info_url_requested_cb), NULL);
+g_signal_connect(G_OBJECT(info), "title_changed", G_CALLBACK(poi_info_title_cb), dialog);
+
+s=gtk_html_begin(GTK_HTML(info));
+ls=strlen(p->label);
+
+/* XXX: Format lat/lon according to settings */
+
+WRITE_HTML("<html><meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">" \
+       "<head><title>POI - %s</title></head><body><b>%s (<i>%s</i>)</b><br/><table>" \
+       "<tr><th align=\"left\">Location:</th><td>%.5f, %.5f</td></tr>",
+       ls>0 ? p->label : _("(Unknown)"), 
+       ls>0 ? p->label : _("(Unknown)"), p->cat_label, 
+       p->lat, p->lon);
 
-gtk_html_load_from_string(info, phtml, -1);
+if (p->postal_code)
+       WRITE_HTML("<tr><th align=\"left\">Postal Code:</th><td>%s</td></tr>", p->postal_code);
+if (p->url)
+       WRITE_HTML("<tr><th align=\"left\">Link:</th><td><a href=\"%s\">%s</a></td></tr>", p->url, p->url);
+if (p->desc)
+       WRITE_HTML("<tr><th align=\"left\" colspan=\"2\">Description:</th><td colspan=\"2\">%s</td></tr>", p->desc);
+
+WRITE_HTML("</table><br/>");
+
+# if 0
+if (p->image)
+       WRITE_HTML("<img src=\"%s\">", p->image);
+#endif
+
+WRITE_HTML("</body></html>");
+gtk_html_end(GTK_HTML(info), s, GTK_HTML_STREAM_OK);
+
+poi_free(p);
 
 gtk_window_set_default_size(GTK_WINDOW(dialog), 500, 250);
 gtk_widget_show_all(dialog);
-gtk_dialog_run(dialog);
+gtk_dialog_run(GTK_DIALOG(dialog));
 gtk_widget_destroy(dialog);
 
-g_free(phtml);
-poi_free(p);
-
 return TRUE;
 }
+
 /**
  * poi_edit_dialog
  *
@@ -935,45 +970,122 @@ return TRUE;
 }
 
 static gboolean
-poi_quick_button_cb(GtkWidget *w, gpointer data)
+poi_quick_button_cb(GtkWidget *button, gpointer data)
 {
 poi_info *p;
+poi_quick_data *qpdata=(poi_quick_data *)data;
+
+g_assert(data);
 
 p=poi_new();
-p->cat_id=GPOINTER_TO_INT(data);
+p->cat_id=GPOINTER_TO_INT(g_object_get_data(G_OBJECT(button), "pid"));
 
-if (strlen(gtk_entry_get_text(GTK_ENTRY(qp.label)))>0) {
-       p->label=g_strdup(gtk_entry_get_text(GTK_ENTRY(qp.label)));
+if (strlen(gtk_entry_get_text(GTK_ENTRY(qpdata->label)))>0) {
+       p->label=g_strdup(gtk_entry_get_text(GTK_ENTRY(qpdata->label)));
 } else {
        p->label=g_strdup("");
 }
 
-p->lat=qp.lat;
-p->lon=qp.lon;
+if (qpdata->fixed==TRUE) {
+       p->lat=qpdata->lat;
+       p->lon=qpdata->lon;
+} else {
+       p->lat=_gps->data.lat;  
+       p->lon=_gps->data.lon;
+}
 p->desc=g_strdup("Quick POI, update information please.");
 
 /* poi_add frees the label and desc so strdup */
 if (poi_add(p)==FALSE) {
        popup_error(_window, _("Problem adding POI"));
 } else {
+       gchar *txt;
+
        map_poi_cache_clear();
+
+       /* Add a text break to the current track */
+       /* XXX: Get category string in here if label is empty */
+       txt=g_strdup_printf("QP(%d): %f %f %s", p->cat_id, p->lat, p->lon, p->label);
+       track_insert_mark_text(txt);
+
        map_force_redraw();
-       gtk_widget_destroy(qp.dialog);
+       if (qpdata->close==TRUE)
+               gtk_widget_destroy(qpdata->dialog);
+
+       hildon_banner_show_information(_window, NULL, _("POI added"));
 }
 
 poi_free(p);
 return TRUE;
 }
 
-gboolean
-poi_quick_dialog(gdouble lat, gdouble lon)
+static void
+poi_button_set_icon(GtkWidget *button, node_type_t t)
 {
-GtkWidget *table;
+const gchar *iname;
+GdkPixbuf *icon=NULL;
+
+iname=poi_get_icon_from_type(t);
+if (iname)
+       icon=poi_get_icon(iname, TRUE);
+if (icon) {
+       gtk_button_set_image(GTK_BUTTON(button), gtk_image_new_from_pixbuf(icon));
+       gtk_button_set_image_position(GTK_BUTTON(button), GTK_POS_TOP);
+}
+}
+
+/**
+ * Return a vbox filled with POI_QUICK_BUTTONS POI buttons and one for other POI, 
+ * with an optinal POI label.
+ */
+GtkWidget *
+poi_quick_button_box(poi_quick_data *qpdata)
+{
+GtkWidget *table, *vbox;
 GtkWidget *buttons[POI_QUICK_BUTTONS];
-guint x,y;
+GtkWidget *otherbtn;
+guint x, y;
 
-qp.lat=lat;
-qp.lon=lon;
+vbox=gtk_vbox_new(FALSE, 6);
+table = gtk_table_new(QPBS_Y, QPBS_X, TRUE);
+gtk_table_set_col_spacings(GTK_TABLE(table), 6);
+gtk_table_set_row_spacings(GTK_TABLE(table), 6);
+
+for (y=0;y<QPBS_Y;y++) {
+       for (x=0;x<QPBS_X;x++) {
+               guint p=y*(QPBS_X)+x;
+
+               g_debug("B: %d (%d,%d)", p, x, y);
+               buttons[p]=gtk_button_new_with_label(quick_poi_categories[p].name);
+               poi_button_set_icon(buttons[p], quick_poi_categories[p].type);
+               gtk_table_attach(GTK_TABLE(table), buttons[p], x, x+1, y, y+1, GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 6, 6);
+               g_object_set_data(G_OBJECT(buttons[p]), "pid", GINT_TO_POINTER(quick_poi_categories[p].type));
+               g_signal_connect(G_OBJECT(buttons[p]), "clicked", G_CALLBACK(poi_quick_button_cb), qpdata);
+       }
+}
+
+gtk_box_pack_start(GTK_BOX(vbox), table, TRUE, TRUE, 0);
+
+/* Other POI button */
+otherbtn=gtk_button_new_with_label("Other POI");
+poi_button_set_icon(otherbtn, NODE_AMENITY_GENERIC);
+gtk_box_pack_start(GTK_BOX(vbox), otherbtn, TRUE, FALSE, 6);
+g_object_set_data(G_OBJECT(otherbtn), "pid", GINT_TO_POINTER(NODE_AMENITY_GENERIC));
+g_signal_connect(G_OBJECT(otherbtn), "clicked", G_CALLBACK(poi_quick_button_cb), qpdata);
+
+gtk_box_pack_start(GTK_BOX(vbox), qpdata->label = gtk_entry_new(), TRUE, FALSE, 0);
+
+return vbox;
+}
+
+/**
+ * Show simple dialog for adding a POI.
+ * Uses the above helper to get the buttons. The location is given as paramters.
+ */
+gboolean
+poi_quick_dialog(gdouble lat, gdouble lon)
+{
+GtkWidget *box;
 
 qp.dialog=gtk_dialog_new_with_buttons(_("Quick POI"),
                GTK_WINDOW(_window),
@@ -984,21 +1096,14 @@ qp.dialog=gtk_dialog_new_with_buttons(_("Quick POI"),
 
 help_dialog_help_enable(GTK_DIALOG(qp.dialog), HELP_ID_POIQUICK);
 
-gtk_box_pack_start(GTK_BOX(GTK_DIALOG(qp.dialog)->vbox), table = gtk_table_new(3, 3, FALSE), TRUE, TRUE, 0);
-
-gtk_table_set_col_spacings(GTK_TABLE(table), 6);
-gtk_table_set_row_spacings(GTK_TABLE(table), 6);
-gtk_table_set_homogeneous(GTK_TABLE(table), TRUE);
+qp.fixed=TRUE;
+qp.lat=lat;
+qp.lon=lon;
+qp.close=TRUE;
 
-for (x=1;x<=3;x++) {
-       for (y=1;y<=3;y++) {
-               buttons[x*y]=gtk_button_new_with_label(quick_poi_categories[x*y-1].name);
-               gtk_table_attach(GTK_TABLE(table), buttons[x*y], x-1, x, y-1, y, GTK_FILL, 0, 2, 4);
-               g_signal_connect(G_OBJECT(buttons[x*y]), "clicked", G_CALLBACK(poi_quick_button_cb), GINT_TO_POINTER(quick_poi_categories[x*y-1].type));
-       }
-}
+box=poi_quick_button_box(&qp);
+gtk_box_pack_start(GTK_BOX(GTK_DIALOG(qp.dialog)->vbox), box, TRUE, TRUE, 0);
 
-gtk_box_pack_start(GTK_BOX(GTK_DIALOG(qp.dialog)->vbox), qp.label = gtk_entry_new(), TRUE, TRUE, 0);
 gtk_widget_show_all(qp.dialog);
 
 if (gtk_dialog_run(GTK_DIALOG(qp.dialog))==GTK_RESPONSE_REJECT)