]> git.0d.be Git - empathy.git/blobdiff - libempathy-gtk/empathy-search-bar.c
local-xmpp-assistant-widget: increase row-spacing
[empathy.git] / libempathy-gtk / empathy-search-bar.c
index e4bad3b4321fc2c3f81808bdda28b06fb1d42253..d23380d4d84c7670a8ca957c1d303b656b777e36 100644 (file)
  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
  */
 
+#include "config.h"
+
 #include <glib.h>
 #include <glib-object.h>
+#include <glib/gi18n-lib.h>
 #include <gtk/gtk.h>
+#include <gdk/gdkkeysyms.h>
 
 #include <libempathy/empathy-utils.h>
 
@@ -29,7 +33,7 @@
 
 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathySearchBar)
 
-G_DEFINE_TYPE (EmpathySearchBar, empathy_search_bar, GTK_TYPE_BIN);
+G_DEFINE_TYPE (EmpathySearchBar, empathy_search_bar, GTK_TYPE_BOX);
 
 typedef struct _EmpathySearchBarPriv EmpathySearchBarPriv;
 struct _EmpathySearchBarPriv
@@ -37,7 +41,10 @@ struct _EmpathySearchBarPriv
   EmpathyChatView *chat_view;
 
   GtkWidget *search_entry;
-  gchar *last_search;
+
+  GtkWidget *search_match_case;
+
+  GtkWidget *search_match_case_toolitem;
 
   GtkWidget *search_close;
   GtkWidget *search_previous;
@@ -56,77 +63,43 @@ empathy_search_bar_new (EmpathyChatView *view)
 }
 
 static void
-empathy_search_bar_size_request (GtkWidget *widget,
-    GtkRequisition *requisition)
-{
-  GtkBin *bin;
-  GtkWidget *child;
-
-  bin = GTK_BIN (widget);
-  child = gtk_bin_get_child (bin);
-
-  if (child && gtk_widget_get_visible (child))
-    {
-      GtkRequisition child_requisition;
-
-      gtk_widget_size_request (child, &child_requisition);
-
-      requisition->width = child_requisition.width;
-      requisition->height = child_requisition.height;
-    }
-}
-
-static void
-empathy_search_bar_size_allocate (GtkWidget *widget,
-    GtkAllocation *allocation)
+empathy_search_bar_update_buttons (EmpathySearchBar *self,
+    gchar *search,
+    gboolean match_case)
 {
-  GtkBin *bin;
-  GtkWidget *child;
-  GtkAllocation child_allocation;
-
-  bin = GTK_BIN (widget);
-  child = gtk_bin_get_child (bin);
+  gboolean can_go_forward = FALSE;
+  gboolean can_go_backward = FALSE;
 
-  gtk_widget_set_allocation (widget, allocation);
+  EmpathySearchBarPriv* priv = GET_PRIV (self);
 
-  if (child && gtk_widget_get_visible (child))
-    {
-      child_allocation.x = allocation->x;
-      child_allocation.y = allocation->y;
-      child_allocation.width = MAX (allocation->width, 0);
-      child_allocation.height = MAX (allocation->height, 0);
+  /* update previous / next buttons */
+  empathy_chat_view_find_abilities (priv->chat_view, search, match_case,
+      &can_go_backward, &can_go_forward);
 
-      gtk_widget_size_allocate (child, &child_allocation);
-    }
+  gtk_widget_set_sensitive (priv->search_previous,
+      can_go_backward && !EMP_STR_EMPTY (search));
+  gtk_widget_set_sensitive (priv->search_next,
+      can_go_forward && !EMP_STR_EMPTY (search));
 }
 
 static void
-empathy_search_bar_finalize (GObject *object)
+empathy_search_bar_update (EmpathySearchBar *self)
 {
-  EmpathySearchBarPriv *priv = GET_PRIV (object);
-
-  g_free (priv->last_search);
-}
+  gchar *search;
+  gboolean match_case;
+  EmpathySearchBarPriv *priv = GET_PRIV (self);
 
-static void
-empathy_search_bar_update_buttons (EmpathySearchBar *self)
-{
-  gboolean can_go_forward = FALSE;
-  gboolean can_go_backward = FALSE;
+  search = gtk_editable_get_chars (GTK_EDITABLE(priv->search_entry), 0, -1);
+  match_case = gtk_toggle_button_get_active (
+      GTK_TOGGLE_BUTTON (priv->search_match_case));
 
-  EmpathySearchBarPriv* priv = GET_PRIV (self);
+  /* highlight & search */
+  empathy_chat_view_highlight (priv->chat_view, search, match_case);
 
-  /* update previous / next buttons */
-  if (priv->last_search)
-    {
-      empathy_chat_view_find_abilities (priv->chat_view, priv->last_search,
-          &can_go_backward, &can_go_forward);
-    }
+  /* update the buttons */
+  empathy_search_bar_update_buttons (self, search, match_case);
 
-  gtk_widget_set_sensitive (priv->search_previous,
-      can_go_backward && !EMP_STR_EMPTY (priv->last_search));
-  gtk_widget_set_sensitive (priv->search_next,
-      can_go_forward && !EMP_STR_EMPTY (priv->last_search));
+  g_free (search);
 }
 
 void
@@ -134,15 +107,8 @@ empathy_search_bar_show (EmpathySearchBar *self)
 {
   EmpathySearchBarPriv *priv = GET_PRIV (self);
 
-  if (priv->last_search)
-    {
-      /* make sure we have an up to date last_search */
-      g_free (priv->last_search);
-      priv->last_search = gtk_editable_get_chars (
-          GTK_EDITABLE (priv->search_entry), 0, -1);
-    }
-  empathy_chat_view_highlight (priv->chat_view, priv->last_search);
-  empathy_search_bar_update_buttons (self);
+  /* update the highlighting and buttons */
+  empathy_search_bar_update (self);
 
   /* grab the focus to the search entry */
   gtk_widget_grab_focus (priv->search_entry);
@@ -150,62 +116,143 @@ empathy_search_bar_show (EmpathySearchBar *self)
   gtk_widget_show (GTK_WIDGET (self));
 }
 
-
-static void
-empathy_search_bar_close_cb (GtkButton *button,
-    gpointer user_data)
+void
+empathy_search_bar_hide (EmpathySearchBar *self)
 {
-  EmpathySearchBarPriv *priv = GET_PRIV (user_data);
+  EmpathySearchBarPriv *priv = GET_PRIV (self);
 
-  empathy_chat_view_highlight (priv->chat_view, "");
-  gtk_widget_hide (GTK_WIDGET (user_data));
+  empathy_chat_view_highlight (priv->chat_view, "", FALSE);
+  gtk_widget_hide (GTK_WIDGET (self));
 
   /* give the focus back to the focus-chain with the chat view */
   gtk_widget_grab_focus (GTK_WIDGET (priv->chat_view));
 }
 
 static void
-empathy_search_bar_entry_changed (GtkEditable *entry,
-    gpointer user_data)
+empathy_search_bar_search (EmpathySearchBar *self,
+    gboolean next,
+    gboolean new_search)
 {
-  EmpathySearchBarPriv *priv;
-  gboolean found;
   gchar *search;
+  gboolean found;
+  gboolean match_case;
+  EmpathySearchBarPriv *priv;
 
-  priv = GET_PRIV (user_data);
+  priv = GET_PRIV (self);
 
-  search = gtk_editable_get_chars (entry, 0, -1);
+  search = gtk_editable_get_chars (GTK_EDITABLE(priv->search_entry), 0, -1);
+  match_case = gtk_toggle_button_get_active (
+      GTK_TOGGLE_BUTTON (priv->search_match_case));
 
-  found = empathy_chat_view_find_previous (priv->chat_view, search, TRUE);
+  /* highlight & search */
+  empathy_chat_view_highlight (priv->chat_view, search, match_case);
+  if (next)
+    {
+      found = empathy_chat_view_find_next (priv->chat_view,
+          search,
+          new_search,
+          match_case);
+    }
+  else
+    {
+      found = empathy_chat_view_find_previous (priv->chat_view,
+          search,
+          new_search,
+          match_case);
+    }
+
+  /* (don't) display the not found label */
   gtk_widget_set_visible (priv->search_not_found,
       !(found || EMP_STR_EMPTY (search)));
 
-  empathy_chat_view_highlight (priv->chat_view, search);
+  /* update the buttons */
+  empathy_search_bar_update_buttons (self, search, match_case);
 
-  g_free (priv->last_search);
-  priv->last_search = search;
+  g_free (search);
+}
 
-  empathy_search_bar_update_buttons (EMPATHY_SEARCH_BAR (user_data));
+static void
+empathy_search_bar_close_cb (GtkButton *button,
+    gpointer user_data)
+{
+  empathy_search_bar_hide (EMPATHY_SEARCH_BAR (user_data));
 }
 
 static void
-empathy_search_bar_next_cb (GtkButton *button,
+empathy_search_bar_entry_changed (GtkEditable *entry,
     gpointer user_data)
 {
-  EmpathySearchBarPriv *priv = GET_PRIV (user_data);
+  empathy_search_bar_search (EMPATHY_SEARCH_BAR (user_data), FALSE, TRUE);
+}
 
-  empathy_chat_view_find_next (priv->chat_view, priv->last_search, FALSE);
-  empathy_search_bar_update_buttons (EMPATHY_SEARCH_BAR (user_data));
+static gboolean
+empathy_search_bar_key_pressed (GtkWidget   *widget,
+    GdkEventKey *event,
+    gpointer user_data)
+{
+  if (event->keyval == GDK_KEY_Escape)
+    {
+      empathy_search_bar_hide (EMPATHY_SEARCH_BAR (widget));
+      return TRUE;
+    }
+  return FALSE;
+}
+
+static void
+empathy_search_bar_next_cb (GtkButton *button,
+    gpointer user_data)
+{
+  empathy_search_bar_search (EMPATHY_SEARCH_BAR (user_data), TRUE, FALSE);
 }
 
 static void
 empathy_search_bar_previous_cb (GtkButton *button,
     gpointer user_data)
 {
-  EmpathySearchBarPriv *priv = GET_PRIV (user_data);
+  empathy_search_bar_search (EMPATHY_SEARCH_BAR (user_data), FALSE, FALSE);
+}
+
+static void
+empathy_search_bar_match_case_toggled (GtkButton *button,
+    gpointer user_data)
+{
+  empathy_search_bar_update (EMPATHY_SEARCH_BAR (user_data));
+}
 
-  empathy_chat_view_find_previous (priv->chat_view, priv->last_search, FALSE);
-  empathy_search_bar_update_buttons (EMPATHY_SEARCH_BAR (user_data));
+static void
+empathy_search_bar_match_case_menu_toggled (GtkWidget *check,
+    gpointer user_data)
+{
+  EmpathySearchBarPriv* priv = GET_PRIV ( EMPATHY_SEARCH_BAR (user_data));
+  gboolean match_case;
+
+  match_case = gtk_check_menu_item_get_active (GTK_CHECK_MENU_ITEM (check));
+
+  gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (priv->search_match_case),
+      match_case);
+}
+
+static gboolean
+empathy_searchbar_create_menu_proxy_cb (GtkToolItem *toolitem,
+    gpointer user_data)
+{
+  EmpathySearchBarPriv* priv = GET_PRIV ( EMPATHY_SEARCH_BAR (user_data));
+  GtkWidget *checkbox_menu;
+  gboolean match_case;
+
+  checkbox_menu = gtk_check_menu_item_new_with_mnemonic (_("_Match case"));
+  match_case = gtk_toggle_button_get_active (
+      GTK_TOGGLE_BUTTON (priv->search_match_case));
+  gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (checkbox_menu),
+      match_case);
+
+  g_signal_connect (checkbox_menu, "toggled",
+      G_CALLBACK (empathy_search_bar_match_case_menu_toggled), user_data);
+
+  gtk_tool_item_set_proxy_menu_item (toolitem, "menu-proxy",
+      checkbox_menu);
+
+  return TRUE;
 }
 
 static void
@@ -229,6 +276,7 @@ empathy_search_bar_init (EmpathySearchBar * self)
       "search_previous", &priv->search_previous,
       "search_next", &priv->search_next,
       "search_not_found", &priv->search_not_found,
+      "search_match_case", &priv->search_match_case,
       NULL);
   g_free (filename);
 
@@ -238,9 +286,14 @@ empathy_search_bar_init (EmpathySearchBar * self)
       "search_entry", "changed", empathy_search_bar_entry_changed,
       "search_previous", "clicked", empathy_search_bar_previous_cb,
       "search_next", "clicked", empathy_search_bar_next_cb,
+      "search_match_case", "toggled", empathy_search_bar_match_case_toggled,
+      "search_match_case_toolitem", "create-menu-proxy", empathy_searchbar_create_menu_proxy_cb,
       NULL);
 
-  gtk_container_add (GTK_CONTAINER (self), internal);
+  g_signal_connect (G_OBJECT (self), "key-press-event",
+      G_CALLBACK (empathy_search_bar_key_pressed), NULL);
+
+  gtk_box_pack_start (GTK_BOX (self), internal, TRUE, TRUE, 0);
   gtk_widget_show_all (internal);
   gtk_widget_hide (priv->search_not_found);
   g_object_unref (gui);
@@ -250,14 +303,14 @@ static void
 empathy_search_bar_class_init (EmpathySearchBarClass *class)
 {
   GObjectClass *gobject_class = G_OBJECT_CLASS (class);
-  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (class);
-
-  gobject_class->finalize = empathy_search_bar_finalize;
 
   g_type_class_add_private (gobject_class, sizeof (EmpathySearchBarPriv));
-
-  /* Neither GtkBin nor GtkContainer seems to do this for us :( */
-  widget_class->size_request = empathy_search_bar_size_request;
-  widget_class->size_allocate = empathy_search_bar_size_allocate;
 }
 
+void
+empathy_search_bar_paste_clipboard (EmpathySearchBar *self)
+{
+  EmpathySearchBarPriv *priv = GET_PRIV (self);
+
+  gtk_editable_paste_clipboard (GTK_EDITABLE (priv->search_entry));
+}