/* * Copyright (C) 2004 Tommi Komulainen * Copyright (C) 2004, 2005 Christian Persch * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * $Id$ */ #include "config.h" #include "ephy-find-toolbar.h" #include "ephy-embed-find.h" #include "ephy-embed-factory.h" #include "ephy-debug.h" #include <gdk/gdkkeysyms.h> #include <glib/gi18n.h> #include <gtk/gtkarrow.h> #include <gtk/gtkhbox.h> #include <gtk/gtkimage.h> #include <gtk/gtkentry.h> #include <gtk/gtklabel.h> #include <gtk/gtkseparatortoolitem.h> #include <gtk/gtkstock.h> #include <gtk/gtktoolbutton.h> #include <gtk/gtkalignment.h> #include <gtk/gtkmain.h> #include <string.h> #undef FIND_WHILE_TYPING_IN_EMBED #define EPHY_FIND_TOOLBAR_GET_PRIVATE(object) (G_TYPE_INSTANCE_GET_PRIVATE ((object),EPHY_TYPE_FIND_TOOLBAR, EphyFindToolbarPrivate)) struct _EphyFindToolbarPrivate { EphyEmbedFind *find; EphyWindow *window; EphyEmbed *embed; GtkWidget *entry; GtkToolItem *next; GtkToolItem *prev; GtkToolItem *sep; GtkToolItem *status_item; GtkWidget *status_label; gulong set_focus_handler; guint preedit_changed : 1; guint prevent_activate : 1; guint activated : 1; guint explicit_focus : 1; guint links_only : 1; guint typing_ahead : 1; }; enum { PROP_0, PROP_WINDOW }; enum { NEXT, PREVIOUS, CLOSE, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; static GObjectClass *parent_class = NULL; /* private functions */ static EphyEmbedFind * get_find (EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; if (priv->find == NULL) { LOG ("Creating the finder now"); priv->find = EPHY_EMBED_FIND (ephy_embed_factory_new_object (EPHY_TYPE_EMBED_FIND)); g_return_val_if_fail (priv->embed == NULL || GTK_WIDGET_REALIZED (GTK_WIDGET (priv->embed)), priv->find); ephy_embed_find_set_embed (priv->find, priv->embed); } return priv->find; } static void set_controls (EphyFindToolbar *toolbar, gboolean can_find_next, gboolean can_find_prev) { EphyFindToolbarPrivate *priv = toolbar->priv; gtk_widget_set_sensitive (GTK_WIDGET (priv->next), can_find_next); gtk_widget_set_sensitive (GTK_WIDGET (priv->prev), can_find_prev); } static void set_status (EphyFindToolbar *toolbar, EphyEmbedFindResult result) { EphyFindToolbarPrivate *priv = toolbar->priv; char *text = NULL; switch (result) { case EPHY_EMBED_FIND_FOUND: text = NULL; break; case EPHY_EMBED_FIND_NOTFOUND: text = _("Phrase not found"); break; case EPHY_EMBED_FIND_FOUNDWRAPPED: text = _("Find wrapped"); break; } gtk_label_set_text (GTK_LABEL (priv->status_label), text != NULL ? text : ""); g_object_set (priv->sep, "visible", text != NULL, NULL); g_object_set (priv->status_item, "visible", text != NULL, NULL); } static void clear_status (EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; gtk_widget_hide (GTK_WIDGET (priv->sep)); gtk_widget_hide (GTK_WIDGET (priv->status_item)); } static void tab_content_changed_cb (EphyEmbed *embed, const char *uri, EphyFindToolbar *toolbar) { set_controls (toolbar, TRUE, TRUE); } #ifdef HAVE_TYPEAHEADFIND #ifdef FIND_WHILE_TYPING_IN_EMBED /* Cut and paste from gtkwindow.c */ static void send_focus_change (GtkWidget *widget, gboolean in) { GdkEvent *event; event = gdk_event_new (GDK_FOCUS_CHANGE); g_object_ref (widget); if (in) { GTK_WIDGET_SET_FLAGS (widget, GTK_HAS_FOCUS); } else { GTK_WIDGET_UNSET_FLAGS (widget, GTK_HAS_FOCUS); } event->focus_change.type = GDK_FOCUS_CHANGE; event->focus_change.window = (GdkWindow *) g_object_ref (widget->window); event->focus_change.in = in; gtk_widget_event (widget, event); g_object_notify (G_OBJECT (widget), "has_focus"); g_object_unref (widget); gdk_event_free (event); } #endif /* FIND_WHILE_TYPING_IN_EMBED */ /* Code adapted from gtktreeview.c:gtk_tree_view_key_press() and * gtk_tree_view_real_start_interactive_seach() */ static gboolean tab_search_key_press_cb (EphyEmbed *embed, GdkEventKey *event, EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; GtkWidget *widget = (GtkWidget *) toolbar; #ifdef FIND_WHILE_TYPING_IN_EMBED GtkEntry *entry = (GtkEntry *) priv->entry; GdkWindow *event_window; gboolean retval = FALSE; guint oldhash, newhash; #endif g_return_val_if_fail (event != NULL, FALSE); /* don't do anything in PPV mode */ if (ephy_window_get_is_print_preview (priv->window)) return FALSE; /* check for / and ' which open the find toolbar in text resp. link mode */ if (GTK_WIDGET_VISIBLE (widget) == FALSE) { if (event->keyval == GDK_slash) { ephy_find_toolbar_open (toolbar, FALSE, TRUE); return TRUE; } else if (event->keyval == GDK_apostrophe) { ephy_find_toolbar_open (toolbar, TRUE, TRUE); return TRUE; } } return FALSE; #ifdef FIND_WHILE_TYPING_IN_EMBED /* don't do anything if the find toolbar is hidden */ if (GTK_WIDGET_VISIBLE (widget) == FALSE || event->keyval == GDK_Return || event->keyval == GDK_KP_Enter || event->keyval == GDK_ISO_Enter) { return FALSE; } oldhash = g_str_hash (gtk_entry_get_text (entry)); event_window = event->window; event->window = priv->entry->window; /* Send the event to the window. If the preedit_changed signal is emitted * during this event, we will set priv->imcontext_changed */ priv->preedit_changed = priv->activated = FALSE; priv->prevent_activate = TRUE; retval = gtk_widget_event (priv->entry, (GdkEvent*) event); priv->prevent_activate = FALSE; /* restore event window, else gdk_event_free below will crash */ event->window = event_window; newhash = g_str_hash (gtk_entry_get_text (entry)); /* FIXME: is this correct? */ return retval && (oldhash != newhash || priv->preedit_changed) && !priv->activated; #endif /* FIND_WHILE_TYPING_IN_EMBED */ } #ifdef FIND_WHILE_TYPING_IN_EMBED static gboolean tab_dom_mouse_click_cb (EphyEmbed *embed, gpointer event, EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; if (!priv->explicit_focus) { ephy_find_toolbar_close (toolbar); } return FALSE; } #endif /* FIND_WHILE_TYPING_IN_EMBED */ #endif /* HAVE_TYPEAHEADFIND */ static void find_next_cb (EphyFindToolbar *toolbar) { g_signal_emit (toolbar, signals[NEXT], 0); } static void find_prev_cb (EphyFindToolbar *toolbar) { g_signal_emit (toolbar, signals[PREVIOUS], 0); } static void entry_changed_cb (GtkEntry *entry, EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; const char *text; char *lowercase; EphyEmbedFindResult result; gboolean found = TRUE, case_sensitive; text = gtk_entry_get_text (GTK_ENTRY (priv->entry)); /* Search case-sensitively iff the string includes * non-lowercase character. */ lowercase = g_utf8_strdown (text, -1); case_sensitive = g_utf8_collate (text, lowercase) != 0; g_free (lowercase); ephy_embed_find_set_properties (get_find (toolbar), text, case_sensitive); #ifdef HAVE_TYPEAHEADFIND result = ephy_embed_find_find (get_find (toolbar), text, priv->links_only); found = result == EPHY_EMBED_FIND_FOUND; set_status (toolbar, result); #endif set_controls (toolbar, found, found); } #ifdef FIND_WHILE_TYPING_IN_EMBED static void entry_preedit_changed_cb (GtkIMContext *context, EphyFindToolbar *toolbar) { toolbar->priv->preedit_changed = TRUE; } #endif static gboolean entry_key_press_event_cb (GtkEntry *entry, GdkEventKey *event, EphyFindToolbar *toolbar) { guint mask = gtk_accelerator_get_default_mod_mask (); gboolean handled = FALSE; /* Hide the toolbar when ESC is pressed */ if ((event->state & mask) == 0 && event->keyval == GDK_Escape) { g_signal_emit (toolbar, signals[CLOSE], 0); handled = TRUE; } else if ((event->state & mask) == GDK_CONTROL_MASK && (event->keyval == GDK_Return || event->keyval == GDK_KP_Enter)) { handled = ephy_embed_find_activate_link (get_find (toolbar), event->state); } return handled; } static void entry_activate_cb (GtkWidget *entry, EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; #ifdef FIND_WHILE_TYPING_IN_EMBED priv->activated = TRUE; if (priv->prevent_activate) return; #endif if (priv->typing_ahead) { ephy_embed_find_activate_link (get_find (toolbar), 0); } else { g_signal_emit (toolbar, signals[NEXT], 0); } } static void set_focus_cb (EphyWindow *window, GtkWidget *widget, EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; GtkWidget *wtoolbar = GTK_WIDGET (toolbar); while (widget != NULL && widget != wtoolbar) { widget = widget->parent; } /* if widget == toolbar, the new focus widget is in the toolbar */ if (widget == wtoolbar) { priv->explicit_focus = TRUE; } else if (priv->explicit_focus) { g_signal_emit (toolbar, signals[CLOSE], 0); } } static void sync_print_preview_mode (EphyWindow *window, GParamSpec *pspec, EphyFindToolbar *toolbar) { if (ephy_window_get_is_print_preview (window) && GTK_WIDGET_VISIBLE (GTK_WIDGET (toolbar))) { ephy_find_toolbar_close (toolbar); } } static void ephy_find_toolbar_set_window (EphyFindToolbar *toolbar, EphyWindow *window) { EphyFindToolbarPrivate *priv = toolbar->priv; priv->window = window; sync_print_preview_mode (window, NULL, toolbar); g_signal_connect (window, "notify::print-preview-mode", G_CALLBACK (sync_print_preview_mode), toolbar); } static void ephy_find_toolbar_parent_set (GtkWidget *widget, GtkWidget *previous_parent) { EphyFindToolbar *toolbar = EPHY_FIND_TOOLBAR (widget); EphyFindToolbarPrivate *priv = toolbar->priv; GtkWidget *toplevel; if (widget->parent != NULL && priv->set_focus_handler == 0) { toplevel = gtk_widget_get_toplevel (widget); priv->set_focus_handler = g_signal_connect (toplevel, "set-focus", G_CALLBACK (set_focus_cb), toolbar); } } static void ephy_find_toolbar_grab_focus (GtkWidget *widget) { EphyFindToolbar *toolbar = EPHY_FIND_TOOLBAR (widget); EphyFindToolbarPrivate *priv = toolbar->priv; gtk_widget_grab_focus (GTK_WIDGET (priv->entry)); g_return_if_fail (priv->explicit_focus); } static void ephy_find_toolbar_init (EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv; GtkToolbar *gtoolbar; GtkToolItem *item; GtkWidget *alignment, *arrow, *box, *label; priv = toolbar->priv = EPHY_FIND_TOOLBAR_GET_PRIVATE (toolbar); gtoolbar = GTK_TOOLBAR (toolbar); gtk_toolbar_set_style (gtoolbar, GTK_TOOLBAR_BOTH_HORIZ); /* Find: |_____| */ alignment = gtk_alignment_new (0.0, 0.5, 1.0, 0.0); gtk_alignment_set_padding (GTK_ALIGNMENT (alignment), 0, 0, 2, 2); box = gtk_hbox_new (FALSE, 12); gtk_container_add (GTK_CONTAINER (alignment), box); label = gtk_label_new (NULL); gtk_label_set_markup (GTK_LABEL (label), _("Find:")); gtk_box_pack_start (GTK_BOX (box), label, FALSE, FALSE, 0); priv->entry = gtk_entry_new (); gtk_entry_set_width_chars (GTK_ENTRY (priv->entry), 32); gtk_entry_set_max_length (GTK_ENTRY (priv->entry), 512); gtk_box_pack_start (GTK_BOX (box), priv->entry, TRUE, TRUE, 0); item = gtk_tool_item_new (); gtk_container_add (GTK_CONTAINER (item), alignment); //gtk_tool_item_set_expand (item, TRUE); gtk_toolbar_insert (GTK_TOOLBAR (toolbar), item, -1); gtk_widget_show_all (GTK_WIDGET (item)); /* Prev */ arrow = gtk_arrow_new (GTK_ARROW_LEFT, GTK_SHADOW_NONE); label = gtk_label_new (_("Find Previous")); priv->prev = gtk_tool_button_new (arrow, _("Find Previous")); gtk_tool_item_set_is_important (priv->prev, TRUE); gtk_tool_item_set_tooltip (priv->prev, gtoolbar->tooltips, _("Find previous occurrence of the search string"), NULL); gtk_toolbar_insert (GTK_TOOLBAR (toolbar), priv->prev, -1); gtk_widget_show_all (GTK_WIDGET (priv->prev)); /* Next */ arrow = gtk_arrow_new (GTK_ARROW_RIGHT, GTK_SHADOW_NONE); label = gtk_label_new (_("Find Next")); priv->next = gtk_tool_button_new (arrow, _("Find Next")); gtk_tool_item_set_is_important (priv->next, TRUE); gtk_tool_item_set_tooltip (priv->next, gtoolbar->tooltips, _("Find next occurrence of the search string"), NULL); gtk_toolbar_insert (GTK_TOOLBAR (toolbar), priv->next, -1); gtk_widget_show_all (GTK_WIDGET (priv->next)); priv->sep = gtk_separator_tool_item_new (); gtk_toolbar_insert (GTK_TOOLBAR (toolbar), priv->sep, -1); priv->status_item = gtk_tool_item_new (); gtk_tool_item_set_expand (priv->status_item, TRUE); priv->status_label = gtk_label_new (""); gtk_misc_set_alignment (GTK_MISC (priv->status_label), 0.0, 0.5); gtk_label_set_ellipsize (GTK_LABEL (priv->status_label), PANGO_ELLIPSIZE_END); gtk_container_add (GTK_CONTAINER (priv->status_item), priv->status_label); gtk_widget_show (priv->status_label); gtk_toolbar_insert (GTK_TOOLBAR (toolbar), priv->status_item, -1); /* connect signals */ g_signal_connect (priv->entry, "key-press-event", G_CALLBACK (entry_key_press_event_cb), toolbar); g_signal_connect_after (priv->entry, "changed", G_CALLBACK (entry_changed_cb), toolbar); #ifdef FIND_WHILE_TYPING_IN_EMBED g_signal_connect (GTK_ENTRY (priv->entry)->im_context, "preedit-changed", G_CALLBACK (entry_preedit_changed_cb), toolbar); #endif g_signal_connect (priv->entry, "activate", G_CALLBACK (entry_activate_cb), toolbar); g_signal_connect_swapped (priv->next, "clicked", G_CALLBACK (find_next_cb), toolbar); g_signal_connect_swapped (priv->prev, "clicked", G_CALLBACK (find_prev_cb), toolbar); } static void ephy_find_toolbar_finalize (GObject *object) { EphyFindToolbar *toolbar = EPHY_FIND_TOOLBAR (object); EphyFindToolbarPrivate *priv = toolbar->priv; if (priv->find != NULL) { g_object_unref (priv->find); } parent_class->finalize (object); } static void ephy_find_toolbar_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { /* no readable properties */ g_assert_not_reached (); } static void ephy_find_toolbar_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { EphyFindToolbar *toolbar = EPHY_FIND_TOOLBAR (object); switch (prop_id) { case PROP_WINDOW: ephy_find_toolbar_set_window (toolbar, (EphyWindow *) g_value_get_object (value)); break; } } static void ephy_find_toolbar_class_init (EphyFindToolbarClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); parent_class = g_type_class_peek_parent (klass); object_class->finalize = ephy_find_toolbar_finalize; object_class->get_property = ephy_find_toolbar_get_property; object_class->set_property = ephy_find_toolbar_set_property; widget_class->parent_set = ephy_find_toolbar_parent_set; widget_class->grab_focus = ephy_find_toolbar_grab_focus; klass->next = ephy_find_toolbar_find_next; klass->previous = ephy_find_toolbar_find_previous; klass->close = ephy_find_toolbar_close; signals[NEXT] = g_signal_new ("next", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (EphyFindToolbarClass, next), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[PREVIOUS] = g_signal_new ("previous", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (EphyFindToolbarClass, previous), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[CLOSE] = g_signal_new ("close", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (EphyFindToolbarClass, close), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); g_object_class_install_property (object_class, PROP_WINDOW, g_param_spec_object ("window", "Window", "Parent window", EPHY_TYPE_WINDOW, (GParamFlags) (G_PARAM_WRITABLE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB | G_PARAM_CONSTRUCT_ONLY))); g_type_class_add_private (klass, sizeof (EphyFindToolbarPrivate)); } /* public functions */ GType ephy_find_toolbar_get_type (void) { static GType type = 0; if (G_UNLIKELY (type == 0)) { static const GTypeInfo our_info = { sizeof (EphyFindToolbarClass), NULL, /* base_init */ NULL, /* base_finalize */ (GClassInitFunc) ephy_find_toolbar_class_init, NULL, NULL, /* class_data */ sizeof (EphyFindToolbar), 0, /* n_preallocs */ (GInstanceInitFunc) ephy_find_toolbar_init }; type = g_type_register_static (GTK_TYPE_TOOLBAR, "EphyFindToolbar", &our_info, 0); } return type; } EphyFindToolbar * ephy_find_toolbar_new (EphyWindow *window) { return EPHY_FIND_TOOLBAR (g_object_new (EPHY_TYPE_FIND_TOOLBAR, "window", window, NULL)); } const char * ephy_find_toolbar_get_text (EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; return gtk_entry_get_text (GTK_ENTRY (priv->entry)); } void ephy_find_toolbar_set_embed (EphyFindToolbar *toolbar, EphyEmbed *embed) { EphyFindToolbarPrivate *priv = toolbar->priv; if (priv->embed == embed) return; if (priv->embed != NULL) { g_signal_handlers_disconnect_matched (embed, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, toolbar); } priv->embed = embed; if (embed != NULL) { set_controls (toolbar, TRUE, TRUE); g_signal_connect_object (embed, "ge-content-change", G_CALLBACK (tab_content_changed_cb), toolbar, G_CONNECT_AFTER); #ifdef HAVE_TYPEAHEADFIND g_signal_connect_object (embed, "ge-search-key-press", G_CALLBACK (tab_search_key_press_cb), toolbar, 0); #ifdef FIND_WHILE_TYPING_IN_EMBED g_signal_connect_object (embed, "dom-mouse-click", G_CALLBACK (tab_dom_mouse_click_cb), toolbar, 0); #endif /* FIND_WHILE_TYPING_IN_EMBED */ #endif /* HAVE_TYPEAHEADFIND */ if (priv->find != NULL) { g_return_if_fail (GTK_WIDGET_REALIZED (GTK_WIDGET (priv->embed))); ephy_embed_find_set_embed (priv->find, embed); } } } void ephy_find_toolbar_find_next (EphyFindToolbar *toolbar) { EphyEmbedFindResult result; gboolean found; result = ephy_embed_find_find_again (get_find (toolbar), TRUE); found = result == EPHY_EMBED_FIND_FOUND; set_controls (toolbar, found, found); set_status (toolbar, result); } void ephy_find_toolbar_find_previous (EphyFindToolbar *toolbar) { EphyEmbedFindResult result; gboolean found; result = ephy_embed_find_find_again (get_find (toolbar), FALSE); found = result == EPHY_EMBED_FIND_FOUND; set_controls (toolbar, found, found); set_status (toolbar, result); } void ephy_find_toolbar_open (EphyFindToolbar *toolbar, gboolean links_only, gboolean typing_ahead) { EphyFindToolbarPrivate *priv = toolbar->priv; gboolean clear_search = typing_ahead; g_return_if_fail (priv->embed != NULL); priv->typing_ahead = typing_ahead; priv->links_only = links_only; priv->explicit_focus = FALSE; clear_status (toolbar); if (clear_search) { gtk_entry_set_text (GTK_ENTRY (priv->entry), ""); } else { gtk_editable_select_region (GTK_EDITABLE (priv->entry), 0, -1); } gtk_widget_show (GTK_WIDGET (toolbar)); #if defined(HAVE_TYPEAHEADFIND) && defined(FIND_WHILE_TYPING_IN_EMBED) gtk_widget_grab_focus (GTK_WIDGET (priv->embed)); send_focus_change (priv->entry, TRUE); if (strlen(gtk_entry_get_text (GTK_ENTRY (priv->entry))) != 0) { entry_changed_cb (GTK_ENTRY (priv->entry), toolbar); } #else gtk_widget_grab_focus (GTK_WIDGET (toolbar)); #endif } void ephy_find_toolbar_close (EphyFindToolbar *toolbar) { EphyFindToolbarPrivate *priv = toolbar->priv; g_return_if_fail (priv->embed != NULL); gtk_widget_hide (GTK_WIDGET (toolbar)); /* first unset explicit_focus, else we get infinite recursion */ priv->explicit_focus = FALSE; #ifndef FIND_WHILE_TYPING_IN_EMBED gtk_widget_grab_focus (GTK_WIDGET (priv->embed)); #endif }