diff options
-rw-r--r-- | src/empathy-map-view.c | 228 | ||||
-rw-r--r-- | src/empathy-map-view.h | 24 | ||||
-rw-r--r-- | src/empathy-map-view.ui | 123 |
3 files changed, 221 insertions, 154 deletions
diff --git a/src/empathy-map-view.c b/src/empathy-map-view.c index d266f6587..31fa997c2 100644 --- a/src/empathy-map-view.c +++ b/src/empathy-map-view.c @@ -43,10 +43,13 @@ #define DEBUG_FLAG EMPATHY_DEBUG_LOCATION #include <libempathy/empathy-debug.h> -typedef struct { +G_DEFINE_TYPE (EmpathyMapView, empathy_map_view, GTK_TYPE_WINDOW); + +#define GET_PRIV(self) ((EmpathyMapViewPriv *)((EmpathyMapView *) self)->priv) + +struct _EmpathyMapViewPriv { EmpathyContactList *contact_list; - GtkWidget *window; GtkWidget *zoom_in; GtkWidget *zoom_out; GtkWidget *throbber; @@ -56,25 +59,26 @@ typedef struct { /* reffed (EmpathyContact *) => borrowed (ChamplainMarker *) */ GHashTable *markers; gulong members_changed_id; -} EmpathyMapView; +}; static void map_view_state_changed (ChamplainView *view, GParamSpec *gobject, - EmpathyMapView *window) + EmpathyMapView *self) { ChamplainState state; + EmpathyMapViewPriv *priv = GET_PRIV (self); g_object_get (G_OBJECT (view), "state", &state, NULL); if (state == CHAMPLAIN_STATE_LOADING) { - gtk_spinner_start (GTK_SPINNER (window->throbber)); - gtk_widget_show (window->throbber); + gtk_spinner_start (GTK_SPINNER (priv->throbber)); + gtk_widget_show (priv->throbber); } else { - gtk_spinner_stop (GTK_SPINNER (window->throbber)); - gtk_widget_hide (window->throbber); + gtk_spinner_stop (GTK_SPINNER (priv->throbber)); + gtk_widget_hide (priv->throbber); } } @@ -98,6 +102,7 @@ static void map_view_update_contact_position (EmpathyMapView *self, EmpathyContact *contact) { + EmpathyMapViewPriv *priv = GET_PRIV (self); gdouble lon, lat; GValue *value; GHashTable *location; @@ -106,7 +111,7 @@ map_view_update_contact_position (EmpathyMapView *self, has_location = contact_has_location (contact); - marker = g_hash_table_lookup (self->markers, contact); + marker = g_hash_table_lookup (priv->markers, contact); if (marker == NULL) { if (!has_location) @@ -152,33 +157,38 @@ map_view_contact_location_notify (EmpathyContact *contact, static void map_view_zoom_in_cb (GtkWidget *widget, - EmpathyMapView *window) + EmpathyMapView *self) { - champlain_view_zoom_in (window->map_view); + EmpathyMapViewPriv *priv = GET_PRIV (self); + + champlain_view_zoom_in (priv->map_view); } static void map_view_zoom_out_cb (GtkWidget *widget, - EmpathyMapView *window) + EmpathyMapView *self) { - champlain_view_zoom_out (window->map_view); + EmpathyMapViewPriv *priv = GET_PRIV (self); + + champlain_view_zoom_out (priv->map_view); } static void map_view_zoom_fit_cb (GtkWidget *widget, - EmpathyMapView *window) + EmpathyMapView *self) { + EmpathyMapViewPriv *priv = GET_PRIV (self); GList *item, *children; GPtrArray *markers; - children = clutter_container_get_children (CLUTTER_CONTAINER (window->layer)); + children = clutter_container_get_children (CLUTTER_CONTAINER (priv->layer)); markers = g_ptr_array_sized_new (g_list_length (children) + 1); for (item = children; item != NULL; item = g_list_next (item)) g_ptr_array_add (markers, (gpointer) item->data); g_ptr_array_add (markers, (gpointer) NULL); - champlain_view_ensure_markers_visible (window->map_view, + champlain_view_ensure_markers_visible (priv->map_view, (ChamplainBaseMarker **) markers->pdata, TRUE); @@ -257,9 +267,10 @@ map_view_contacts_update_label (ChamplainMarker *marker) } static ChamplainMarker * -create_marker (EmpathyMapView *window, +create_marker (EmpathyMapView *self, EmpathyContact *contact) { + EmpathyMapViewPriv *priv = GET_PRIV (self); ClutterActor *marker; ClutterActor *texture; GdkPixbuf *avatar; @@ -281,7 +292,7 @@ create_marker (EmpathyMapView *window, g_object_set_data_full (G_OBJECT (marker), "contact", g_object_ref (contact), g_object_unref); - g_hash_table_insert (window->markers, g_object_ref (contact), marker); + g_hash_table_insert (priv->markers, g_object_ref (contact), marker); map_view_contacts_update_label (CHAMPLAIN_MARKER (marker)); @@ -289,7 +300,7 @@ create_marker (EmpathyMapView *window, g_signal_connect (marker, "button-release-event", G_CALLBACK (marker_clicked_cb), contact); - clutter_container_add (CLUTTER_CONTAINER (window->layer), marker, NULL); + clutter_container_add (CLUTTER_CONTAINER (priv->layer), marker, NULL); DEBUG ("Create marker for %s", empathy_contact_get_id (contact)); @@ -297,36 +308,13 @@ create_marker (EmpathyMapView *window, } static void -contact_added (EmpathyMapView *window, +contact_added (EmpathyMapView *self, EmpathyContact *contact) { g_signal_connect (contact, "notify::location", - G_CALLBACK (map_view_contact_location_notify), window); - - map_view_update_contact_position (window, contact); -} - -static void -map_view_destroy_cb (GtkWidget *widget, - EmpathyMapView *window) -{ - GHashTableIter iter; - gpointer contact; + G_CALLBACK (map_view_contact_location_notify), self); - g_source_remove (window->timeout_id); - - g_hash_table_iter_init (&iter, window->markers); - while (g_hash_table_iter_next (&iter, &contact, NULL)) - g_signal_handlers_disconnect_by_func (contact, - map_view_contact_location_notify, window); - - g_signal_handler_disconnect (window->contact_list, - window->members_changed_id); - - g_hash_table_destroy (window->markers); - g_object_unref (window->contact_list); - g_object_unref (window->layer); - g_slice_free (EmpathyMapView, window); + map_view_update_contact_position (self, contact); } static gboolean @@ -345,11 +333,12 @@ map_view_key_press_cb (GtkWidget *widget, } static gboolean -map_view_tick (EmpathyMapView *window) +map_view_tick (EmpathyMapView *self) { + EmpathyMapViewPriv *priv = GET_PRIV (self); GList *marker, *l; - marker = clutter_container_get_children (CLUTTER_CONTAINER (window->layer)); + marker = clutter_container_get_children (CLUTTER_CONTAINER (priv->layer)); for (l = marker; l != NULL; l = g_list_next (l)) map_view_contacts_update_label (l->data); @@ -362,14 +351,15 @@ static void contact_removed (EmpathyMapView *self, EmpathyContact *contact) { + EmpathyMapViewPriv *priv = GET_PRIV (self); ClutterActor *marker; - marker = g_hash_table_lookup (self->markers, contact); + marker = g_hash_table_lookup (priv->markers, contact); if (marker == NULL) return; clutter_actor_destroy (marker); - g_hash_table_remove (self->markers, contact); + g_hash_table_remove (priv->markers, contact); } static void @@ -391,93 +381,155 @@ members_changed_cb (EmpathyContactList *list, } } -GtkWidget * -empathy_map_view_show (void) +static GObject * +empathy_map_view_constructor (GType type, + guint n_construct_params, + GObjectConstructParam *construct_params) +{ + static GObject *window = NULL; + + if (window != NULL) + return window; + + window = G_OBJECT_CLASS (empathy_map_view_parent_class)->constructor ( + type, n_construct_params, construct_params); + + g_object_add_weak_pointer (window, (gpointer) &window); + + return window; +} + +static void +empathy_map_view_finalize (GObject *object) { - static EmpathyMapView *window = NULL; + EmpathyMapViewPriv *priv = GET_PRIV (object); + GHashTableIter iter; + gpointer contact; + + g_source_remove (priv->timeout_id); + + g_hash_table_iter_init (&iter, priv->markers); + while (g_hash_table_iter_next (&iter, &contact, NULL)) + g_signal_handlers_disconnect_by_func (contact, + map_view_contact_location_notify, object); + + g_signal_handler_disconnect (priv->contact_list, + priv->members_changed_id); + + g_hash_table_destroy (priv->markers); + g_object_unref (priv->contact_list); + g_object_unref (priv->layer); + + G_OBJECT_CLASS (empathy_map_view_parent_class)->finalize (object); +} + +static void +empathy_map_view_class_init (EmpathyMapViewClass *klass) +{ + GObjectClass *object_class = G_OBJECT_CLASS (klass); + + object_class->constructor = empathy_map_view_constructor; + object_class->finalize = empathy_map_view_finalize; + + g_type_class_add_private (object_class, sizeof (EmpathyMapViewPriv)); +} + +static void +empathy_map_view_init (EmpathyMapView *self) +{ + EmpathyMapViewPriv *priv; GtkBuilder *gui; GtkWidget *sw; GtkWidget *embed; GtkWidget *throbber_holder; gchar *filename; GList *members, *l; + GtkWidget *main_vbox; - if (window) - { - empathy_window_present (GTK_WINDOW (window->window)); - return window->window; - } + priv = self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, + EMPATHY_TYPE_MAP_VIEW, EmpathyMapViewPriv); - window = g_slice_new0 (EmpathyMapView); + gtk_window_set_title (GTK_WINDOW (self), _("Contact Map View")); + gtk_window_set_role (GTK_WINDOW (self), "map_view"); + gtk_window_set_default_size (GTK_WINDOW (self), 512, 384); + gtk_window_set_position (GTK_WINDOW (self), GTK_WIN_POS_CENTER); /* Set up interface */ filename = empathy_file_lookup ("empathy-map-view.ui", "src"); gui = empathy_builder_get_file (filename, - "map_view", &window->window, - "zoom_in", &window->zoom_in, - "zoom_out", &window->zoom_out, + "main_vbox", &main_vbox, + "zoom_in", &priv->zoom_in, + "zoom_out", &priv->zoom_out, "map_scrolledwindow", &sw, "throbber", &throbber_holder, NULL); g_free (filename); - empathy_builder_connect (gui, window, - "map_view", "destroy", map_view_destroy_cb, - "map_view", "key-press-event", map_view_key_press_cb, + gtk_container_add (GTK_CONTAINER (self), main_vbox); + + empathy_builder_connect (gui, self, "zoom_in", "clicked", map_view_zoom_in_cb, "zoom_out", "clicked", map_view_zoom_out_cb, "zoom_fit", "clicked", map_view_zoom_fit_cb, NULL); - g_object_unref (gui); + g_signal_connect (self, "key-press-event", + G_CALLBACK (map_view_key_press_cb), self); - /* Clear the static pointer to window if the dialog is destroyed */ - g_object_add_weak_pointer (G_OBJECT (window->window), (gpointer *) &window); + g_object_unref (gui); - window->contact_list = EMPATHY_CONTACT_LIST ( + priv->contact_list = EMPATHY_CONTACT_LIST ( empathy_contact_manager_dup_singleton ()); - window->members_changed_id = g_signal_connect (window->contact_list, - "members-changed", G_CALLBACK (members_changed_cb), window); + priv->members_changed_id = g_signal_connect (priv->contact_list, + "members-changed", G_CALLBACK (members_changed_cb), self); - window->throbber = gtk_spinner_new (); - gtk_widget_set_size_request (window->throbber, 16, 16); - gtk_container_add (GTK_CONTAINER (throbber_holder), window->throbber); + priv->throbber = gtk_spinner_new (); + gtk_widget_set_size_request (priv->throbber, 16, 16); + gtk_container_add (GTK_CONTAINER (throbber_holder), priv->throbber); /* Set up map view */ embed = gtk_champlain_embed_new (); - window->map_view = gtk_champlain_embed_get_view (GTK_CHAMPLAIN_EMBED (embed)); - g_object_set (G_OBJECT (window->map_view), "zoom-level", 1, + priv->map_view = gtk_champlain_embed_get_view (GTK_CHAMPLAIN_EMBED (embed)); + g_object_set (G_OBJECT (priv->map_view), "zoom-level", 1, "scroll-mode", CHAMPLAIN_SCROLL_MODE_KINETIC, NULL); - champlain_view_center_on (window->map_view, 36, 0); + champlain_view_center_on (priv->map_view, 36, 0); gtk_container_add (GTK_CONTAINER (sw), embed); gtk_widget_show_all (embed); - window->layer = g_object_ref (champlain_layer_new ()); - champlain_view_add_layer (window->map_view, window->layer); + priv->layer = g_object_ref (champlain_layer_new ()); + champlain_view_add_layer (priv->map_view, priv->layer); - g_signal_connect (window->map_view, "notify::state", - G_CALLBACK (map_view_state_changed), window); + g_signal_connect (priv->map_view, "notify::state", + G_CALLBACK (map_view_state_changed), self); /* Set up contact list. */ - window->markers = g_hash_table_new_full (NULL, NULL, + priv->markers = g_hash_table_new_full (NULL, NULL, (GDestroyNotify) g_object_unref, NULL); members = empathy_contact_list_get_members ( - window->contact_list); + priv->contact_list); for (l = members; l != NULL; l = g_list_next (l)) { - contact_added (window, l->data); + contact_added (self, l->data); g_object_unref (l->data); } g_list_free (members); - empathy_window_present (GTK_WINDOW (window->window)); - /* Set up time updating loop */ - window->timeout_id = g_timeout_add_seconds (5, - (GSourceFunc) map_view_tick, window); + priv->timeout_id = g_timeout_add_seconds (5, + (GSourceFunc) map_view_tick, self); +} + +GtkWidget * +empathy_map_view_show (void) +{ + GtkWidget *window; + + window = g_object_new (EMPATHY_TYPE_MAP_VIEW, NULL); + gtk_widget_show_all (window); + empathy_window_present (GTK_WINDOW (window)); - return window->window; + return window; } diff --git a/src/empathy-map-view.h b/src/empathy-map-view.h index 6628f1e08..3184b8c9e 100644 --- a/src/empathy-map-view.h +++ b/src/empathy-map-view.h @@ -25,6 +25,30 @@ G_BEGIN_DECLS +#define EMPATHY_TYPE_MAP_VIEW (empathy_map_view_get_type ()) +#define EMPATHY_MAP_VIEW(o) (G_TYPE_CHECK_INSTANCE_CAST ((o), EMPATHY_TYPE_MAP_VIEW, EmpathyMapView)) +#define EMPATHY_MAP_VIEW_CLASS(k) (G_TYPE_CHECK_CLASS_CAST ((k), EMPATHY_TYPE_MAP_VIEW, EmpathyMapViewClass)) +#define EMPATHY_IS_MAP_VIEW(o) (G_TYPE_CHECK_INSTANCE_TYPE ((o), EMPATHY_TYPE_MAP_VIEW)) +#define EMPATHY_IS_MAP_VIEW_CLASS(k) (G_TYPE_CHECK_CLASS_TYPE ((k), EMPATHY_TYPE_MAP_VIEW)) +#define EMPATHY_MAP_VIEW_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS ((o), EMPATHY_TYPE_MAP_VIEW, EmpathyMapViewClass)) + +typedef struct _EmpathyMapView EmpathyMapView; +typedef struct _EmpathyMapViewClass EmpathyMapViewClass; +typedef struct _EmpathyMapViewPriv EmpathyMapViewPriv; + +struct _EmpathyMapView +{ + GtkWindow parent; + gpointer priv; +}; + +struct _EmpathyMapViewClass +{ + GtkWindowClass parent_class; +}; + +GType empathy_map_view_get_type (void); + GtkWidget *empathy_map_view_show (void); G_END_DECLS diff --git a/src/empathy-map-view.ui b/src/empathy-map-view.ui index a56090e0e..715de8fa4 100644 --- a/src/empathy-map-view.ui +++ b/src/empathy-map-view.ui @@ -2,97 +2,88 @@ <interface> <requires lib="gtk+" version="2.16"/> <!-- interface-naming-policy toplevel-contextual --> - <object class="GtkWindow" id="map_view"> - <property name="title" translatable="yes">Contact Map View</property> - <property name="role">map_view</property> - <property name="window_position">center</property> - <property name="default_width">512</property> - <property name="default_height">384</property> + <object class="GtkVBox" id="main_vbox"> + <property name="visible">True</property> <child> - <object class="GtkVBox" id="main_vbox"> + <object class="GtkToolbar" id="toolbar"> <property name="visible">True</property> <child> - <object class="GtkToolbar" id="toolbar"> + <object class="GtkToolButton" id="zoom_in"> + <property name="visible">True</property> + <property name="stock_id">gtk-zoom-in</property> + </object> + <packing> + <property name="expand">False</property> + <property name="homogeneous">True</property> + </packing> + </child> + <child> + <object class="GtkToolButton" id="zoom_out"> + <property name="visible">True</property> + <property name="stock_id">gtk-zoom-out</property> + </object> + <packing> + <property name="expand">False</property> + <property name="homogeneous">True</property> + </packing> + </child> + <child> + <object class="GtkToolButton" id="zoom_fit"> + <property name="visible">True</property> + <property name="stock_id">gtk-zoom-fit</property> + </object> + <packing> + <property name="expand">False</property> + <property name="homogeneous">True</property> + </packing> + </child> + <child> + <object class="GtkToolItem" id="toolbutton1"> <property name="visible">True</property> <child> - <object class="GtkToolButton" id="zoom_in"> - <property name="visible">True</property> - <property name="stock_id">gtk-zoom-in</property> - </object> - <packing> - <property name="expand">False</property> - <property name="homogeneous">True</property> - </packing> - </child> - <child> - <object class="GtkToolButton" id="zoom_out"> - <property name="visible">True</property> - <property name="stock_id">gtk-zoom-out</property> - </object> - <packing> - <property name="expand">False</property> - <property name="homogeneous">True</property> - </packing> - </child> - <child> - <object class="GtkToolButton" id="zoom_fit"> - <property name="visible">True</property> - <property name="stock_id">gtk-zoom-fit</property> - </object> - <packing> - <property name="expand">False</property> - <property name="homogeneous">True</property> - </packing> - </child> - <child> - <object class="GtkToolItem" id="toolbutton1"> - <property name="visible">True</property> - <child> - <object class="GtkAlignment" id="alignment1"> - <property name="visible">True</property> - <child> - <placeholder/> - </child> - </object> - </child> - </object> - <packing> - <property name="expand">True</property> - </packing> - </child> - <child> - <object class="GtkToolItem" id="throbber"> + <object class="GtkAlignment" id="alignment1"> <property name="visible">True</property> <child> <placeholder/> </child> </object> - <packing> - <property name="expand">False</property> - </packing> </child> </object> <packing> - <property name="expand">False</property> - <property name="fill">False</property> - <property name="position">0</property> + <property name="expand">True</property> </packing> </child> <child> - <object class="GtkFrame" id="map_scrolledwindow"> + <object class="GtkToolItem" id="throbber"> <property name="visible">True</property> - <property name="resize_mode">queue</property> - <property name="label_xalign">0</property> - <property name="shadow_type">in</property> <child> <placeholder/> </child> </object> <packing> - <property name="position">1</property> + <property name="expand">False</property> </packing> </child> </object> + <packing> + <property name="expand">False</property> + <property name="fill">False</property> + <property name="position">0</property> + </packing> + </child> + <child> + <object class="GtkFrame" id="map_scrolledwindow"> + <property name="visible">True</property> + <property name="resize_mode">queue</property> + <property name="label_xalign">0</property> + <property name="shadow_type">in</property> + <child> + <placeholder/> + </child> + </object> + <packing> + <property name="position">1</property> + </packing> </child> </object> </interface> |