/*
* Copyright (C) 2002 Ricardo Fernández Pascual
* Copyright (C) 2003 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, 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 <glib-object.h>
#include <gtk/gtkhbox.h>
#include <gtk/gtkentry.h>
#include <gtk/gtkwindow.h>
#include <gdk/gdkkeysyms.h>
#include <gtk/gtkmain.h>
#include <gtk/gtktooltips.h>
#include <libgnomeui/gnome-entry.h>
#include <string.h>
/**
* Private data
*/
struct _EphyLocationEntryPrivate {
GtkWidget *combo;
GtkWidget *entry;
char *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;
char *autocompletion_key;
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_construct_contents (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 GObjectClass *parent_class = NULL;
/**
* Signals enums and ids
*/
enum EphyLocationEntrySignalsEnum {
ACTIVATED,
FINISHED,
USER_CHANGED,
LAST_SIGNAL
};
static gint EphyLocationEntrySignals[LAST_SIGNAL];
#define MENU_ID "ephy-location-entry-menu-id"
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 (EGG_TYPE_TOOL_ITEM,
"EphyLocationEntry",
&our_info, 0);
}
return ephy_location_entry_type;
}
static gboolean
ephy_location_entry_set_tooltip (EggToolItem *tool_item,
GtkTooltips *tooltips,
const char *tip_text,
const char *tip_private)
{
EphyLocationEntry *entry = EPHY_LOCATION_ENTRY (tool_item);
g_return_val_if_fail (EPHY_IS_LOCATION_ENTRY (entry), FALSE);
gtk_tooltips_set_tip (tooltips, entry->priv->entry, tip_text, tip_private);
return TRUE;
}
static void
ephy_location_entry_class_init (EphyLocationEntryClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
EggToolItemClass *tool_item_class = EGG_TOOL_ITEM_CLASS (klass);
parent_class = g_type_class_peek_parent (klass);
object_class->finalize = ephy_location_entry_finalize_impl;
tool_item_class->set_tooltip = ephy_location_entry_set_tooltip;
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;
LOG ("EphyLocationEntry initialising %p", w)
p = g_new0 (EphyLocationEntryPrivate, 1);
w->priv = p;
p->last_action_target = NULL;
p->before_completion = NULL;
p->user_changed = TRUE;
p->autocompletion_key = NULL;
ephy_location_entry_construct_contents (w);
egg_tool_item_set_expand (EGG_TOOL_ITEM (w), TRUE);
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->autocompletion_key);
g_free (p);
G_OBJECT_CLASS (parent_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_construct_contents (EphyLocationEntry *w)
{
EphyLocationEntryPrivate *p = w->priv;
GtkWidget *hbox, *list;
LOG ("EphyLocationEntry constructing contents %p", w)
hbox = gtk_hbox_new (FALSE, 0);
gtk_container_add (GTK_CONTAINER (w), hbox);
gtk_widget_show (hbox);
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 (hbox), 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;
/* Reset it because if we do a grab on click (like when pasting
text in the location entry, the entry will lose the release
event and will not reset it. It's what gtk does for the entry
popup */
GTK_ENTRY (p->entry)->button = 0;
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));
}