/* * Copyright (C) 2002 Ricardo Fernández Pascual * * 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, 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. */ #include "ephy-location-entry.h" #include "ephy-autocompletion-window.h" #include "ephy-marshal.h" #include "eel-gconf-extensions.h" #include "ephy-prefs.h" #include "ephy-debug.h" #include #include #include #include #include #include /** * Private data */ struct _EphyLocationEntryPrivate { GtkWidget *combo; GtkWidget *entry; gchar *before_completion; EphyAutocompletion *autocompletion; EphyAutocompletionWindow *autocompletion_window; gboolean autocompletion_window_visible; gint autocompletion_timeout; gint show_alternatives_timeout; gboolean block_set_autocompletion_key; gboolean going_to_site; gboolean user_changed; gchar *autocompletion_key; gchar *last_completion; char *last_action_target; }; #define AUTOCOMPLETION_DELAY 10 #define SHOW_ALTERNATIVES_DELAY 100 /** * Private functions, only availble from this file */ static void ephy_location_entry_class_init (EphyLocationEntryClass *klass); static void ephy_location_entry_init (EphyLocationEntry *w); static void ephy_location_entry_finalize_impl (GObject *o); static void ephy_location_entry_build (EphyLocationEntry *w); static gboolean ephy_location_entry_key_press_event_cb (GtkWidget *entry, GdkEventKey *event, EphyLocationEntry *w); static void ephy_location_entry_activate_cb (GtkEntry *entry, EphyLocationEntry *w); static void ephy_location_entry_autocompletion_sources_changed_cb (EphyAutocompletion *aw, EphyLocationEntry *w); static gint ephy_location_entry_autocompletion_show_alternatives_to (EphyLocationEntry *w); static void ephy_location_entry_autocompletion_window_url_activated_cb /***/ (EphyAutocompletionWindow *aw, const gchar *target, int action, EphyLocationEntry *w); static void ephy_location_entry_list_event_after_cb (GtkWidget *list, GdkEvent *event, EphyLocationEntry *e); static void ephy_location_entry_editable_changed_cb (GtkEditable *editable, EphyLocationEntry *e); static void ephy_location_entry_set_autocompletion_key (EphyLocationEntry *e); static void ephy_location_entry_autocompletion_show_alternatives (EphyLocationEntry *w); static void ephy_location_entry_autocompletion_hide_alternatives (EphyLocationEntry *w); static void ephy_location_entry_autocompletion_window_hidden_cb (EphyAutocompletionWindow *aw, EphyLocationEntry *w); static void insert_text_cb (GtkWidget *editable, char *new_text, int new_text_length, int *position, EphyLocationEntry *w); static void delete_text_cb (GtkWidget *editable, int start_pos, int end_pos, EphyLocationEntry *w); static gpointer gtk_hbox_class; /** * Signals enums and ids */ enum EphyLocationEntrySignalsEnum { ACTIVATED, FINISHED, USER_CHANGED, LAST_SIGNAL }; static gint EphyLocationEntrySignals[LAST_SIGNAL]; GType ephy_location_entry_get_type (void) { static GType ephy_location_entry_type = 0; if (ephy_location_entry_type == 0) { static const GTypeInfo our_info = { sizeof (EphyLocationEntryClass), NULL, NULL, (GClassInitFunc) ephy_location_entry_class_init, NULL, NULL, sizeof (EphyLocationEntry), 0, (GInstanceInitFunc) ephy_location_entry_init }; ephy_location_entry_type = g_type_register_static (GTK_TYPE_HBOX, "EphyLocationEntry", &our_info, 0); } return ephy_location_entry_type; } static void ephy_location_entry_class_init (EphyLocationEntryClass *klass) { G_OBJECT_CLASS (klass)->finalize = ephy_location_entry_finalize_impl; gtk_hbox_class = g_type_class_peek_parent (klass); EphyLocationEntrySignals[ACTIVATED] = g_signal_new ( "activated", G_OBJECT_CLASS_TYPE (klass), G_SIGNAL_RUN_FIRST | G_SIGNAL_RUN_LAST | G_SIGNAL_RUN_CLEANUP, G_STRUCT_OFFSET (EphyLocationEntryClass, activated), NULL, NULL, ephy_marshal_VOID__STRING_STRING, G_TYPE_NONE, 2, G_TYPE_STRING, G_TYPE_STRING); EphyLocationEntrySignals[FINISHED] = g_signal_new ( "finished", G_OBJECT_CLASS_TYPE (klass), G_SIGNAL_RUN_FIRST | G_SIGNAL_RUN_LAST | G_SIGNAL_RUN_CLEANUP, G_STRUCT_OFFSET (EphyLocationEntryClass, finished), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0, G_TYPE_NONE); EphyLocationEntrySignals[USER_CHANGED] = g_signal_new ( "user_changed", G_OBJECT_CLASS_TYPE (klass), G_SIGNAL_RUN_FIRST | G_SIGNAL_RUN_LAST | G_SIGNAL_RUN_CLEANUP, G_STRUCT_OFFSET (EphyLocationEntryClass, user_changed), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0, G_TYPE_NONE); } static gboolean location_focus_out_cb (GtkWidget *widget, GdkEventFocus *event, EphyLocationEntry *w) { g_signal_emit (w, EphyLocationEntrySignals[FINISHED], 0); return FALSE; } static void ephy_location_entry_init (EphyLocationEntry *w) { EphyLocationEntryPrivate *p = g_new0 (EphyLocationEntryPrivate, 1); w->priv = p; p->last_action_target = NULL; p->before_completion = NULL; p->user_changed = TRUE; ephy_location_entry_build (w); g_signal_connect (w->priv->entry, "focus_out_event", G_CALLBACK (location_focus_out_cb), w); } static void ephy_location_entry_finalize_impl (GObject *o) { EphyLocationEntry *w = EPHY_LOCATION_ENTRY (o); EphyLocationEntryPrivate *p = w->priv; if (p->autocompletion) { g_signal_handlers_disconnect_matched (p->autocompletion, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, w); g_signal_handlers_disconnect_matched (p->autocompletion_window, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, w); g_object_unref (G_OBJECT (p->autocompletion)); g_object_unref (G_OBJECT (p->autocompletion_window)); } LOG ("EphyLocationEntry finalized") g_free (p->before_completion); g_free (p); G_OBJECT_CLASS (gtk_hbox_class)->finalize (o); } GtkWidget * ephy_location_entry_new (void) { return GTK_WIDGET (g_object_new (EPHY_TYPE_LOCATION_ENTRY, NULL)); } static gboolean ephy_location_entry_button_press_event_cb (GtkWidget *entry, GdkEventButton *event, EphyLocationEntry *w) { if (event->button == 1 && event->type == GDK_2BUTTON_PRESS) { ephy_location_entry_activate (w); return TRUE; } return FALSE; } static void ephy_location_entry_build (EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; GtkWidget *list; p->combo = gnome_entry_new ("ephy-url-history"); p->entry = GTK_COMBO (p->combo)->entry; gtk_widget_show (p->combo); gtk_box_pack_start (GTK_BOX (w), p->combo, TRUE, TRUE, 0); g_signal_connect (p->entry, "key-press-event", G_CALLBACK (ephy_location_entry_key_press_event_cb), w); g_signal_connect (p->entry, "insert-text", G_CALLBACK (insert_text_cb), w); g_signal_connect (p->entry, "delete-text", G_CALLBACK (delete_text_cb), w); g_signal_connect (p->entry, "button-press-event", G_CALLBACK (ephy_location_entry_button_press_event_cb), w); g_signal_connect (p->entry, "activate", G_CALLBACK (ephy_location_entry_activate_cb), w); g_signal_connect (p->entry, "changed", G_CALLBACK (ephy_location_entry_editable_changed_cb), w); list = GTK_COMBO (p->combo)->list; g_signal_connect_after (list, "event-after", G_CALLBACK (ephy_location_entry_list_event_after_cb), w); } static gboolean ephy_location_ignore_prefix (EphyLocationEntry *w) { char *text; int text_len; int i, k; gboolean result = FALSE; static const gchar *prefixes[] = { EPHY_AUTOCOMPLETION_USUAL_WEB_PREFIXES, NULL }; text = ephy_location_entry_get_location (w); text_len = g_utf8_strlen (text, -1); for (i = 0; prefixes[i] != NULL; i++) { const char *prefix = prefixes[i]; for (k = 0; k < g_utf8_strlen (prefix, -1); k++) { if (text_len == (k + 1) && (strncmp (text, prefix, k + 1) == 0)) { result = TRUE; } } } g_free (text); return result; } static gint ephy_location_entry_autocompletion_show_alternatives_to (EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; g_free (p->before_completion); p->before_completion = gtk_editable_get_chars (GTK_EDITABLE (p->entry), 0, -1); if (ephy_location_ignore_prefix (w)) return FALSE; if (p->autocompletion) { LOG ("Show alternatives") ephy_location_entry_set_autocompletion_key (w); ephy_location_entry_autocompletion_show_alternatives (w); } p->show_alternatives_timeout = 0; return FALSE; } static void ephy_location_entry_autocompletion_hide_alternatives (EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; if (p->autocompletion_window) { ephy_autocompletion_window_hide (p->autocompletion_window); p->autocompletion_window_visible = FALSE; } } static void ephy_location_entry_autocompletion_show_alternatives (EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; if (p->autocompletion_window) { ephy_autocompletion_window_show (p->autocompletion_window); p->autocompletion_window_visible = TRUE; } } static void ephy_location_entry_autocompletion_unselect_alternatives (EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; if (p->autocompletion_window) { ephy_autocompletion_window_unselect (p->autocompletion_window); } } static int get_editable_number_of_chars (GtkEditable *editable) { char *text; int length; text = gtk_editable_get_chars (editable, 0, -1); length = g_utf8_strlen (text, -1); g_free (text); return length; } static gboolean position_is_at_end (GtkEditable *editable) { int end; end = get_editable_number_of_chars (editable); return gtk_editable_get_position (editable) == end; } static void delete_text_cb (GtkWidget *editable, int start_pos, int end_pos, EphyLocationEntry *w) { ephy_location_entry_autocompletion_hide_alternatives (w); } static void insert_text_cb (GtkWidget *editable, char *new_text, int new_text_length, int *position, EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; GtkWidget *window; window = gtk_widget_get_toplevel (editable); g_return_if_fail (window != NULL); if (!GTK_WINDOW (window)->has_focus) return; if (p->going_to_site) return; if (p->autocompletion_timeout != 0) { g_source_remove (p->autocompletion_timeout); p->autocompletion_timeout = 0; } if (p->show_alternatives_timeout != 0) { g_source_remove (p->show_alternatives_timeout); p->show_alternatives_timeout = 0; } ephy_location_entry_autocompletion_unselect_alternatives (w); if (position_is_at_end (GTK_EDITABLE (editable))) { p->show_alternatives_timeout = g_timeout_add (SHOW_ALTERNATIVES_DELAY, (GSourceFunc) ephy_location_entry_autocompletion_show_alternatives_to, w); } } static gboolean ephy_location_entry_key_press_event_cb (GtkWidget *entry, GdkEventKey *event, EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; switch (event->keyval) { case GDK_Left: case GDK_Right: ephy_location_entry_autocompletion_hide_alternatives (w); return FALSE; case GDK_Escape: ephy_location_entry_set_location (w, p->before_completion); gtk_editable_set_position (GTK_EDITABLE (p->entry), -1); ephy_location_entry_autocompletion_hide_alternatives (w); return FALSE; default: break; } return FALSE; } static gboolean ephy_location_entry_content_is_text (const char *content) { return ((g_strrstr (content, ".") == NULL) && (g_strrstr (content, "/") == NULL) && (g_strrstr (content, ":") == NULL)); } static void ephy_location_entry_activate_cb (GtkEntry *entry, EphyLocationEntry *w) { char *content; char *target = NULL; content = gtk_editable_get_chars (GTK_EDITABLE(entry), 0, -1); if (w->priv->last_action_target && ephy_location_entry_content_is_text (content)) { target = g_strdup (w->priv->last_action_target); } else { target = content; content = NULL; g_free ( w->priv->last_action_target); w->priv->last_action_target = NULL; } ephy_location_entry_autocompletion_hide_alternatives (w); LOG ("In ephy_location_entry_activate_cb, activating %s", content) g_signal_emit (w, EphyLocationEntrySignals[ACTIVATED], 0, content, target); g_signal_emit (w, EphyLocationEntrySignals[FINISHED], 0); g_free (content); g_free (target); } static void ephy_location_entry_autocompletion_sources_changed_cb (EphyAutocompletion *aw, EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; LOG ("in ephy_location_entry_autocompletion_sources_changed_cb") if (p->show_alternatives_timeout == 0 && p->autocompletion_window_visible) { p->show_alternatives_timeout = g_timeout_add (SHOW_ALTERNATIVES_DELAY, (GSourceFunc) ephy_location_entry_autocompletion_show_alternatives_to, w); } } void ephy_location_entry_set_location (EphyLocationEntry *w, const gchar *new_location) { EphyLocationEntryPrivate *p = w->priv; int pos; p->user_changed = FALSE; g_signal_handlers_block_by_func (G_OBJECT (p->entry), delete_text_cb, w); gtk_editable_delete_text (GTK_EDITABLE (p->entry), 0, -1); g_signal_handlers_unblock_by_func (G_OBJECT (p->entry), delete_text_cb, w); g_signal_handlers_block_by_func (G_OBJECT (p->entry), insert_text_cb, w); gtk_editable_insert_text (GTK_EDITABLE (p->entry), new_location, strlen(new_location), &pos); g_signal_handlers_unblock_by_func (G_OBJECT (p->entry), insert_text_cb, w); p->user_changed = TRUE; } gchar * ephy_location_entry_get_location (EphyLocationEntry *w) { char *location = gtk_editable_get_chars (GTK_EDITABLE (w->priv->entry), 0, -1); return location; } static void ephy_location_entry_autocompletion_window_url_selected_cb (EphyAutocompletionWindow *aw, const char *target, int action, EphyLocationEntry *w) { if (target) { ephy_location_entry_set_location (w, action ? w->priv->before_completion : target); } else { ephy_location_entry_set_location (w, w->priv->before_completion); } gtk_editable_set_position (GTK_EDITABLE (w->priv->entry), -1); } void ephy_location_entry_set_autocompletion (EphyLocationEntry *w, EphyAutocompletion *ac) { EphyLocationEntryPrivate *p = w->priv; if (p->autocompletion) { g_signal_handlers_disconnect_matched (p->autocompletion, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, w); g_signal_handlers_disconnect_matched (p->autocompletion_window, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, w); g_object_unref (G_OBJECT (p->autocompletion)); g_object_unref (p->autocompletion_window); } p->autocompletion = ac; if (p->autocompletion) { g_object_ref (G_OBJECT (p->autocompletion)); p->autocompletion_window = ephy_autocompletion_window_new (p->autocompletion, p->entry); g_signal_connect (p->autocompletion_window, "activated", G_CALLBACK (ephy_location_entry_autocompletion_window_url_activated_cb), w); g_signal_connect (p->autocompletion_window, "selected", G_CALLBACK (ephy_location_entry_autocompletion_window_url_selected_cb), w); g_signal_connect (p->autocompletion_window, "hidden", G_CALLBACK (ephy_location_entry_autocompletion_window_hidden_cb), w); g_signal_connect (p->autocompletion, "sources-changed", G_CALLBACK (ephy_location_entry_autocompletion_sources_changed_cb), w); ephy_location_entry_set_autocompletion_key (w); } } static void ephy_location_entry_autocompletion_window_url_activated_cb (EphyAutocompletionWindow *aw, const char *target, int action, EphyLocationEntry *w) { char *content; if (action) { if (w->priv->last_action_target) g_free (w->priv->last_action_target); w->priv->last_action_target = g_strdup (target); } else { ephy_location_entry_set_location (w, target); } content = gtk_editable_get_chars (GTK_EDITABLE(w->priv->entry), 0, -1); LOG ("In location_entry_autocompletion_window_url_activated_cb, going to %s", content); ephy_location_entry_autocompletion_hide_alternatives (w); g_signal_emit (w, EphyLocationEntrySignals[ACTIVATED], 0, action ? content : NULL, target); g_free (content); } static void ephy_location_entry_autocompletion_window_hidden_cb (EphyAutocompletionWindow *aw, EphyLocationEntry *w) { EphyLocationEntryPrivate *p = w->priv; LOG ("In location_entry_autocompletion_window_hidden_cb"); p->autocompletion_window_visible = FALSE; if (p->show_alternatives_timeout) { g_source_remove (p->show_alternatives_timeout); p->show_alternatives_timeout = 0; } if (p->autocompletion_timeout) { g_source_remove (p->autocompletion_timeout); p->autocompletion_timeout = 0; } } void ephy_location_entry_activate (EphyLocationEntry *w) { GtkWidget *toplevel; toplevel = gtk_widget_get_toplevel (w->priv->entry); gtk_editable_select_region (GTK_EDITABLE(w->priv->entry), 0, -1); gtk_window_set_focus (GTK_WINDOW(toplevel), w->priv->entry); } static void ephy_location_entry_list_event_after_cb (GtkWidget *list, GdkEvent *event, EphyLocationEntry *e) { if (event->type == GDK_BUTTON_PRESS && ((GdkEventButton *) event)->button == 1) { e->priv->going_to_site = TRUE; } } static void ephy_location_entry_editable_changed_cb (GtkEditable *editable, EphyLocationEntry *e) { EphyLocationEntryPrivate *p = e->priv; ephy_location_entry_set_autocompletion_key (e); if (p->going_to_site) { char *url = ephy_location_entry_get_location (e); if (url && url[0] != '\0') { p->going_to_site = FALSE; g_signal_emit (e, EphyLocationEntrySignals[ACTIVATED], 0, NULL, url); g_free (url); } } if (p->user_changed) { g_signal_emit (e, EphyLocationEntrySignals[USER_CHANGED], 0); } } static void ephy_location_entry_set_autocompletion_key (EphyLocationEntry *e) { EphyLocationEntryPrivate *p = e->priv; if (p->autocompletion && !p->block_set_autocompletion_key) { GtkEditable *editable = GTK_EDITABLE (p->entry); gint sstart, send; gchar *text; gtk_editable_get_selection_bounds (editable, &sstart, &send); text = gtk_editable_get_chars (editable, 0, sstart); ephy_autocompletion_set_key (p->autocompletion, text); g_free (p->autocompletion_key); p->autocompletion_key = text; } } void ephy_location_entry_clear_history (EphyLocationEntry *w) { gnome_entry_clear_history (GNOME_ENTRY (w->priv->combo)); }