aboutsummaryrefslogblamecommitdiffstats
path: root/lib/widgets/ephy-location-entry.c
blob: 29df7128ee998758e66c14d662dd3b90bb1f5562 (plain) (tree)























                                                                              
                       







                                   
























































































































                                                                                                          
                                           




                                                      
           

                              
                                                                          







































































                                                                                         
                                         













































                                                                               
                                 














                                                                              

                                                                                                                     






                                                                                  
                                                






























                                                                                        




















                                                           









                                                                                                   
                                                            




                                              
                                                               














































































































                                                                                                  

                                                                          































                                                                                                           
                                                                          










                                                                                   
                                                                        
































































































                                                                                                          
                                                                                               










                                                                                  
                                                                          


                                              
                                                                  



































































                                                                                     





                                                                 
/*
 *  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 "ephy-gobject-misc.h"
#include "eel-gconf-extensions.h"
#include "ephy-prefs.h"
#include "ephy-debug.h"

#include <gtk/gtkentry.h>
#include <gtk/gtkwindow.h>
#include <gdk/gdkkeysyms.h>
#include <gtk/gtkmain.h>
#include <libgnomeui/gnome-entry.h>
#include <string.h>

/**
 * 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;

    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_to (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 gpointer gtk_hbox_class;

/**
 * Signals enums and ids
 */
enum EphyLocationEntrySignalsEnum {
    ACTIVATED,
    LAST_SIGNAL
};
static gint EphyLocationEntrySignals[LAST_SIGNAL];

/**
 * EphyLocationEntry object
 */

MAKE_GET_TYPE (ephy_location_entry, "EphyLocationEntry", EphyLocationEntry,
           ephy_location_entry_class_init,
           ephy_location_entry_init, GTK_TYPE_HBOX);

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);
}

static void
ephy_location_entry_init (EphyLocationEntry *w)
{
    EphyLocationEntryPrivate *p = g_new0 (EphyLocationEntryPrivate, 1);
    w->priv = p;
    p->last_action_target = NULL;

    ephy_location_entry_build (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);
    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 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, "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;

    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 gint
ephy_location_entry_autocompletion_to (EphyLocationEntry *w)
{
    EphyLocationEntryPrivate *p = w->priv;
    gchar *text;
    gchar *common_prefix;

    LOG ("Autocompletion to")

    ephy_location_entry_set_autocompletion_key (w);

    {
        GtkEditable *editable = GTK_EDITABLE (p->entry);
        gint sstart, send;
        gint pos = gtk_editable_get_position (editable);
        const gchar *text = gtk_entry_get_text (GTK_ENTRY (p->entry));
        gint text_len = strlen (text);
        gtk_editable_get_selection_bounds (editable, &sstart, &send);

        if (pos != text_len
            || send != text_len)
        {
            /* the user is editing the entry, don't mess it */
            LOG ("The user seems editing the text: pos = %d, strlen (text) = %d, sstart = %d, send = %d",
                 pos, strlen (text), sstart, send)
            p->autocompletion_timeout = 0;
            return FALSE;
        }
    }

    common_prefix = ephy_autocompletion_get_common_prefix (p->autocompletion);

    LOG ("common_prefix: %s", common_prefix)

    if (common_prefix && (!p->before_completion || p->before_completion[0] == '\0'))
    {
        text = ephy_location_entry_get_location (w);
        g_free (p->before_completion);
        p->before_completion = text;
    }

    if (common_prefix)
    {
        /* check original length */
        guint text_len = strlen (p->autocompletion_key);

        p->block_set_autocompletion_key = TRUE;

        /* set entry to completed text */
        gtk_entry_set_text (GTK_ENTRY (p->entry), common_prefix);

        /* move selection appropriately */
        gtk_editable_select_region (GTK_EDITABLE (p->entry), text_len, -1);

        p->block_set_autocompletion_key = FALSE;

        g_free (p->last_completion);
        p->last_completion = common_prefix;
    }

    p->autocompletion_timeout = 0;
    return FALSE;
}

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;
}

/* this is from the old location entry, need to do the autocompletion before implementing this */
static gboolean
ephy_location_entry_key_press_event_cb (GtkWidget *entry, GdkEventKey *event, EphyLocationEntry *w)
{
    EphyLocationEntryPrivate *p = w->priv;
        static gboolean suggest = FALSE;
        guint keyval = event->keyval;

        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;
    }

        /* only suggest heuristic completions if TAB is hit twice */
        if (event->keyval != GDK_Tab)
    {
                suggest = FALSE;
    }

        if (((event->state & GDK_Control_L || event->state & GDK_Control_R) &&
             (keyval == GDK_a || keyval == GDK_b || keyval == GDK_c ||
              keyval == GDK_d || keyval == GDK_e || keyval == GDK_f ||
              keyval == GDK_h || keyval == GDK_k || keyval == GDK_u ||
              keyval == GDK_v || keyval == GDK_w || keyval == GDK_x)) ||
            (event->state == 0 && event->keyval == GDK_BackSpace))
        {
        ephy_location_entry_autocompletion_hide_alternatives (w);
                return FALSE;
        }

        /* don't grab alt combos, thus you can still access the menus. */
        if (event->state & GDK_MOD1_MASK)
        {
        ephy_location_entry_autocompletion_hide_alternatives (w);
                return FALSE;
        }

        /* make sure the end key works at all times */
        if ((!((event->state & GDK_SHIFT_MASK) ||
           (event->state & GDK_CONTROL_MASK) ||
           (event->state & GDK_MOD1_MASK))
         && (event->keyval == GDK_End)))
        {
        ephy_location_entry_autocompletion_hide_alternatives (w);
                gtk_editable_select_region (GTK_EDITABLE (p->entry), 0, 0);
                gtk_editable_set_position (GTK_EDITABLE (p->entry), -1);
        ephy_location_entry_autocompletion_unselect_alternatives (w);
                return TRUE;
        }

    switch (event->keyval)
        {
        case GDK_Left:
        case GDK_Right:
        ephy_location_entry_autocompletion_hide_alternatives (w);
                return FALSE;
        case GDK_Up:
        case GDK_Down:
        case GDK_Page_Up:
        case GDK_Page_Down:
        ephy_location_entry_autocompletion_hide_alternatives (w);
                //ephy_embed_grab_focus (window->active_embed);
                return FALSE;
        case GDK_Tab:
        {
                gchar *common_prefix = NULL;
                gchar *text;

        ephy_location_entry_set_autocompletion_key (w);

                gtk_editable_delete_selection (GTK_EDITABLE (p->entry));
                text = ephy_location_entry_get_location (w);
        ephy_location_entry_autocompletion_unselect_alternatives (w);

        if (p->autocompletion)
        {
            common_prefix = ephy_autocompletion_get_common_prefix (p->autocompletion);
        }
                suggest = FALSE;
                if (common_prefix)
                {
                        if (!p->before_completion)
            {
                                p->before_completion = g_strdup (text);
            }

                        p->block_set_autocompletion_key = TRUE;

            gtk_entry_set_text (GTK_ENTRY (p->entry), common_prefix);
            gtk_editable_set_position (GTK_EDITABLE (p->entry), -1);

                        p->block_set_autocompletion_key = FALSE;

            ephy_location_entry_autocompletion_show_alternatives (w);
                        if (!strcmp (common_prefix, text))
                        {
                                /* really suggest something the next time */
                                suggest = TRUE;
                        }
                        g_free (common_prefix);
                }
                else
                {
            ephy_location_entry_autocompletion_hide_alternatives (w);
                }
        g_free (text);
                return TRUE;
        }
        case GDK_Escape:
        ephy_location_entry_autocompletion_hide_alternatives (w);
                if (p->before_completion)
                {
                        ephy_location_entry_set_location (w, p->before_completion);
                        g_free (p->before_completion);
                        p->before_completion = NULL;
            gtk_editable_set_position (GTK_EDITABLE (p->entry), -1);
                        return TRUE;
                }
                break;
        default:
        ephy_location_entry_autocompletion_unselect_alternatives (w);
        if ((event->string[0] > 32) && (event->string[0] < 126) &&
            position_is_at_end (GTK_EDITABLE (entry)))
                {
                        p->show_alternatives_timeout = g_timeout_add
                                (SHOW_ALTERNATIVES_DELAY,
                 (GSourceFunc) ephy_location_entry_autocompletion_show_alternatives_to, w);
                }
                break;
        }

        return FALSE;
}

static gboolean
ephy_location_entry_content_is_text (const char *content)
{
    return ((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 (ephy_location_entry_content_is_text (content))
    {
        target = w->priv->last_action_target;
    }

    ephy_location_entry_autocompletion_hide_alternatives (w);

    LOG ("In ephy_location_entry_activate_cb, activating %s", content)

    g_signal_emit (w, EphyLocationEntrySignals[ACTIVATED], 0, target, content);
    g_free (content);
}

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->autocompletion_timeout == 0
        && p->last_completion
        && !strcmp (p->last_completion, gtk_entry_get_text (GTK_ENTRY (p->entry))))
    {
        p->autocompletion_timeout = gtk_timeout_add
            (AUTOCOMPLETION_DELAY,
             (GSourceFunc) ephy_location_entry_autocompletion_to, w);
    }

        if (p->show_alternatives_timeout == 0
        && p->autocompletion_window_visible)
    {
        p->show_alternatives_timeout = gtk_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;
    gtk_editable_delete_text (GTK_EDITABLE (p->entry), 0, -1);
    gtk_editable_insert_text (GTK_EDITABLE (p->entry), new_location, g_utf8_strlen (new_location, -1),
                  &pos);
}

gchar *
ephy_location_entry_get_location (EphyLocationEntry *w)
{
    char *location = gtk_editable_get_chars (GTK_EDITABLE (w->priv->entry), 0, -1);
    return location;
}

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, "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)
    {
        gchar *url = ephy_location_entry_get_location (e);
        g_signal_emit
            (e, EphyLocationEntrySignals[ACTIVATED], 0, url);
        g_free (url);
    }
}

static void
ephy_location_entry_editable_changed_cb (GtkEditable *editable, EphyLocationEntry *e)
{
    ephy_location_entry_set_autocompletion_key (e);
}

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));
}