* 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>
#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
EmpathyChatView *chat_view;
GtkWidget *search_entry;
- gchar *last_search;
+
+ GtkWidget *search_match_case;
+
+ GtkWidget *search_match_case_toolitem;
GtkWidget *search_close;
GtkWidget *search_previous;
}
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
{
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);
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
"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);
"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);
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));
+}