[empathy] map-view: be a proper GtkWindow (#625826)



commit 25c5ac2188cde02fe35d9d29eccb11adb6950a8b
Author: Guillaume Desmottes <guillaume desmottes collabora co uk>
Date:   Mon Aug 2 16:40:31 2010 +0200

    map-view: be a proper GtkWindow (#625826)

 src/empathy-map-view.c  |  228 +++++++++++++++++++++++++++++------------------
 src/empathy-map-view.h  |   24 +++++
 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 d266f65..31fa997 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 6628f1e..3184b8c 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 a56090e..715de8f 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>



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]