aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--embed/ephy-browse-history.c5
-rw-r--r--embed/ephy-browse-history.h1
-rw-r--r--lib/widgets/ephy-location-entry.c197
-rw-r--r--src/Makefile.am1
-rw-r--r--src/ephy-completion-model.c950
-rw-r--r--src/ephy-completion-model.h16
-rw-r--r--src/ephy-location-controller.c57
7 files changed, 448 insertions, 779 deletions
diff --git a/embed/ephy-browse-history.c b/embed/ephy-browse-history.c
index 3295f8517..c58eaeaf7 100644
--- a/embed/ephy-browse-history.c
+++ b/embed/ephy-browse-history.c
@@ -181,6 +181,7 @@ ephy_browse_history_get_url (EphyBrowseHistory *history,
void
ephy_browse_history_find_urls (EphyBrowseHistory *history,
gint64 from, gint64 to,
+ guint limit,
GList *substring_list,
EphyHistoryJobCallback callback,
gpointer user_data)
@@ -193,6 +194,10 @@ ephy_browse_history_find_urls (EphyBrowseHistory *history,
query->from = from;
query->to = to;
query->substring_list = substring_list;
+ query->sort_type = EPHY_HISTORY_SORT_MV;
+
+ if (limit != 0)
+ query->limit = limit;
ephy_history_service_query_urls (history->priv->history_service,
query, callback, user_data);
diff --git a/embed/ephy-browse-history.h b/embed/ephy-browse-history.h
index abb61343d..980134af5 100644
--- a/embed/ephy-browse-history.h
+++ b/embed/ephy-browse-history.h
@@ -93,6 +93,7 @@ void ephy_browse_history_get_url (EphyBrowseHistory *history,
void ephy_browse_history_find_urls (EphyBrowseHistory *history,
gint64 from, gint64 to,
+ guint limit,
GList *substring_list,
EphyHistoryJobCallback callback,
gpointer user_data);
diff --git a/lib/widgets/ephy-location-entry.c b/lib/widgets/ephy-location-entry.c
index 24e08d58d..f132c6a56 100644
--- a/lib/widgets/ephy-location-entry.c
+++ b/lib/widgets/ephy-location-entry.c
@@ -51,6 +51,8 @@ struct _EphyLocationEntryPrivate
{
GIcon *lock_gicon;
GdkPixbuf *favicon;
+ GtkTreeModel *model;
+ GtkEntryCompletion *completion;
GSList *search_terms;
@@ -112,17 +114,6 @@ static gint signals[LAST_SIGNAL] = { 0 };
G_DEFINE_TYPE (EphyLocationEntry, ephy_location_entry, GTK_TYPE_ENTRY)
-inline static void
-free_search_terms (GSList *search_terms)
-{
- GSList *iter;
-
- for (iter = search_terms; iter != NULL; iter = iter->next)
- g_regex_unref ((GRegex*)iter->data);
-
- g_slist_free (search_terms);
-}
-
static void
ephy_location_entry_finalize (GObject *object)
{
@@ -141,12 +132,6 @@ ephy_location_entry_finalize (GObject *object)
g_object_unref (priv->lock_gicon);
}
- if (priv->search_terms)
- {
- free_search_terms (priv->search_terms);
- priv->search_terms = NULL;
- }
-
G_OBJECT_CLASS (ephy_location_entry_parent_class)->finalize (object);
}
@@ -278,8 +263,6 @@ editable_changed_cb (GtkEditable *editable,
EphyLocationEntry *entry)
{
EphyLocationEntryPrivate *priv = entry->priv;
- const char *text;
- char *pattern;
update_address_state (entry);
@@ -291,104 +274,6 @@ editable_changed_cb (GtkEditable *editable,
priv->can_redo = FALSE;
}
- if (priv->search_terms)
- {
- free_search_terms (priv->search_terms);
- priv->search_terms = NULL;
- }
-
- text = gtk_entry_get_text (GTK_ENTRY (editable));
-
- /*
- * user is specifying a regular expression, so we will
- * have only one search term
- */
- if (g_str_has_prefix (text, "re:"))
- {
- GRegex *regex;
- pattern = g_strdup (text+3);
- regex = g_regex_new (pattern,
- G_REGEX_CASELESS | G_REGEX_OPTIMIZE,
- G_REGEX_MATCH_NOTEMPTY, NULL);
- priv->search_terms = g_slist_append (priv->search_terms, regex);
- g_free (pattern);
- }
- else
- {
- const char *current;
- const char *ptr;
- char *tmp;
- char *term;
- GRegex *term_regex;
- GRegex *quote_regex;
- gint count;
- gboolean inside_quotes = FALSE;
-
- quote_regex = g_regex_new ("\"", G_REGEX_OPTIMIZE,
- G_REGEX_MATCH_NOTEMPTY, NULL);
-
- /*
- * This code loops through the string using pointer arythmetics.
- * Although the string we are handling may contain UTF-8 chars
- * this works because only ASCII chars affect what is actually
- * copied from the string as a search term.
- */
- for (count = 0, current = ptr = text; ptr[0] != '\0'; ptr++, count++)
- {
- /*
- * If we found a double quote character; we will
- * consume bytes up until the next quote, or
- * end of line;
- */
- if (ptr[0] == '"')
- inside_quotes = !inside_quotes;
-
- /*
- * If we found a space, and we are not looking for a
- * closing double quote, or if the next char is the
- * end of the string, append what we have already as
- * a search term.
- */
- if (((ptr[0] == ' ') && (!inside_quotes)) || ptr[1] == '\0')
- {
- /*
- * We special-case the end of the line because
- * we would otherwise not copy the last character
- * of the search string, since the for loop will
- * stop before that.
- */
- if (ptr[1] == '\0')
- count++;
-
- /*
- * remove quotes, and quote any regex-sensitive
- * characters
- */
- tmp = g_regex_escape_string (current, count);
- term = g_regex_replace (quote_regex, tmp, -1, 0,
- "", G_REGEX_MATCH_NOTEMPTY, NULL);
- g_strstrip (term);
- g_free (tmp);
-
- /* we don't want empty search terms */
- if (term[0] != '\0')
- {
- term_regex = g_regex_new (term,
- G_REGEX_CASELESS | G_REGEX_OPTIMIZE,
- G_REGEX_MATCH_NOTEMPTY, NULL);
- priv->search_terms = g_slist_append (priv->search_terms, term_regex);
- }
- g_free (term);
-
- /* count will be incremented by the for loop */
- count = -1;
- current = ptr + 1;
- }
- }
-
- g_regex_unref (quote_regex);
- }
-
g_signal_emit (entry, signals[USER_CHANGED], 0);
}
@@ -894,7 +779,6 @@ ephy_location_entry_init (EphyLocationEntry *le)
p->user_changed = FALSE;
p->block_update = FALSE;
p->saved_text = NULL;
- p->search_terms = NULL;
p->show_lock = FALSE;
p->dns_prefetch_handler = 0;
@@ -981,18 +865,15 @@ cursor_on_match_cb (GtkEntryCompletion *completion,
static void
textcell_data_func (GtkCellLayout *cell_layout,
- GtkCellRenderer *cell,
- GtkTreeModel *tree_model,
- GtkTreeIter *iter,
- gpointer data)
+ GtkCellRenderer *cell,
+ GtkTreeModel *tree_model,
+ GtkTreeIter *iter,
+ gpointer data)
{
GtkWidget *entry;
EphyLocationEntryPrivate *priv;
PangoAttrList *list;
PangoAttribute *att;
- GMatchInfo *match;
-
- int start, end;
char *ctext;
char *title;
@@ -1038,38 +919,6 @@ textcell_data_func (GtkCellLayout *cell_layout,
ctext = title;
}
- if (priv->search_terms)
- {
- GSList *iter;
- GRegex *regex;
-
- for (iter = priv->search_terms; iter != NULL; iter = iter->next)
- {
- regex = (GRegex*) iter->data;
- g_regex_match (regex, ctext, G_REGEX_MATCH_NOTEMPTY, &match);
-
- while (g_match_info_matches (match))
- {
- g_match_info_fetch_pos (match, 0, &start, &end);
-
- att = pango_attr_weight_new (PANGO_WEIGHT_BOLD);
- att->start_index = start;
- att->end_index = end;
-
- pango_attr_list_insert (list, att);
- g_match_info_next (match, NULL);
- }
-
- g_match_info_free (match);
- match = NULL;
- }
-
- }
-
- g_object_set (cell,
- "attributes", list,
- NULL);
-
g_value_init (&text, G_TYPE_STRING);
g_value_take_string (&text, ctext);
g_object_set_property (G_OBJECT (cell), "text", &text);
@@ -1159,28 +1008,20 @@ ephy_location_entry_set_completion (EphyLocationEntry *entry,
guint extra_col,
guint favicon_col)
{
- GtkTreeModel *sort_model;
GtkEntryCompletion *completion;
GtkCellRenderer *cell;
+ EphyLocationEntryPrivate *priv = entry->priv;
- entry->priv->text_col = text_col;
- entry->priv->action_col = action_col;
- entry->priv->keywords_col = keywords_col;
- entry->priv->relevance_col = relevance_col;
- entry->priv->url_col = url_col;
- entry->priv->extra_col = extra_col;
- entry->priv->favicon_col = favicon_col;
-
- sort_model = gtk_tree_model_sort_new_with_model (model);
-
- gtk_tree_sortable_set_sort_column_id
- (GTK_TREE_SORTABLE (sort_model),
- entry->priv->relevance_col,
- GTK_SORT_DESCENDING);
+ priv->text_col = text_col;
+ priv->action_col = action_col;
+ priv->keywords_col = keywords_col;
+ priv->relevance_col = relevance_col;
+ priv->url_col = url_col;
+ priv->extra_col = extra_col;
+ priv->favicon_col = favicon_col;
completion = gtk_entry_completion_new ();
- gtk_entry_completion_set_model (completion, sort_model);
- g_object_unref (sort_model);
+ gtk_entry_completion_set_model (completion, model);
g_signal_connect (completion, "match-selected",
G_CALLBACK (match_selected_cb), entry);
g_signal_connect_after (completion, "action-activated",
@@ -1213,9 +1054,9 @@ ephy_location_entry_set_completion (EphyLocationEntry *entry,
gtk_cell_renderer_text_set_fixed_height_from_font (GTK_CELL_RENDERER_TEXT (cell), 2);
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (completion),
- cell, textcell_data_func,
- entry,
- NULL);
+ cell, textcell_data_func,
+ entry,
+ NULL);
cell = gtk_cell_renderer_pixbuf_new ();
gtk_cell_layout_pack_end (GTK_CELL_LAYOUT (completion),
@@ -1230,6 +1071,8 @@ ephy_location_entry_set_completion (EphyLocationEntry *entry,
G_CALLBACK (cursor_on_match_cb), entry);
gtk_entry_set_completion (GTK_ENTRY (entry), completion);
+
+ priv->completion = completion;
g_object_unref (completion);
}
diff --git a/src/Makefile.am b/src/Makefile.am
index 70f5574d0..56e699b77 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -92,6 +92,7 @@ libephymain_la_CPPFLAGS = \
-I$(top_srcdir)/embed \
-I$(top_srcdir)/lib \
-I$(top_srcdir)/lib/egg \
+ -I$(top_srcdir)/lib/history \
-I$(top_srcdir)/lib/widgets \
-I$(top_srcdir)/src/bookmarks \
-DEXTENSIONS_DIR=\""$(libdir)/epiphany/$(EPIPHANY_API_VERSION)/extensions"\" \
diff --git a/src/ephy-completion-model.c b/src/ephy-completion-model.c
index ef671850f..0936a0ac3 100644
--- a/src/ephy-completion-model.c
+++ b/src/ephy-completion-model.c
@@ -1,6 +1,6 @@
-/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/*
- * Copyright © 2002 Jorn Baayen <jorn@nl.linux.org>
+ * Copyright © 2012 Igalia S.L.
*
* 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
@@ -21,636 +21,484 @@
#include "config.h"
#include "ephy-completion-model.h"
+#include "ephy-browse-history.h"
+#include "ephy-embed-shell.h"
#include "ephy-favicon-cache.h"
#include "ephy-history.h"
-#include "ephy-node.h"
#include "ephy-shell.h"
+#include "ephy-tree-model-node.h"
#include <string.h>
-static void ephy_completion_model_class_init (EphyCompletionModelClass *klass);
-static void ephy_completion_model_init (EphyCompletionModel *model);
-static void ephy_completion_model_tree_model_init (GtkTreeModelIface *iface);
+G_DEFINE_TYPE (EphyCompletionModel, ephy_completion_model, GTK_TYPE_LIST_STORE)
#define EPHY_COMPLETION_MODEL_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_COMPLETION_MODEL, EphyCompletionModelPrivate))
-struct _EphyCompletionModelPrivate
-{
- EphyHistory *history_service;
- EphyBookmarks *bookmarks_service;
- EphyFaviconCache *favicon_cache;
-
- EphyNode *history;
- EphyNode *bookmarks;
- int stamp;
-};
+struct _EphyCompletionModelPrivate {
+ EphyBrowseHistory *browse_history;
+ EphyHistory *legacy_history_service;
+ EphyFaviconCache *favicon_cache;
-enum
-{
- HISTORY_GROUP,
- BOOKMARKS_GROUP
+ EphyNode *bookmarks;
+ GSList *search_terms;
};
-G_DEFINE_TYPE_WITH_CODE (EphyCompletionModel, ephy_completion_model, G_TYPE_OBJECT,
- G_IMPLEMENT_INTERFACE (GTK_TYPE_TREE_MODEL,
- ephy_completion_model_tree_model_init))
-
static void
-ephy_completion_model_class_init (EphyCompletionModelClass *klass)
-{
- GObjectClass *object_class = G_OBJECT_CLASS (klass);
-
- g_type_class_add_private (object_class, sizeof (EphyCompletionModelPrivate));
-}
-
-static GtkTreePath *
-get_path_real (EphyCompletionModel *model,
- EphyNode *root,
- EphyNode *child)
+ephy_completion_model_constructed (GObject *object)
{
- GtkTreePath *retval;
- int index;
-
- retval = gtk_tree_path_new ();
- index = ephy_node_get_child_index (root, child);
+ GType types[N_COL] = { G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING,
+ G_TYPE_INT, G_TYPE_STRING, G_TYPE_BOOLEAN,
+ GDK_TYPE_PIXBUF };
- if (root == model->priv->bookmarks)
- {
- index += ephy_node_get_n_children (model->priv->history);
- }
-
- gtk_tree_path_append_index (retval, index);
-
- return retval;
+ gtk_list_store_set_column_types (GTK_LIST_STORE (object),
+ N_COL,
+ types);
}
static void
-node_iter_from_node (EphyCompletionModel *model,
- EphyNode *root,
- EphyNode *child,
- GtkTreeIter *iter)
-{
- iter->stamp = model->priv->stamp;
- iter->user_data = child;
- iter->user_data2 = root;
-}
-
-static EphyNode *
-get_index_root (EphyCompletionModel *model, int *index)
-{
- int children;
-
- children = ephy_node_get_n_children (model->priv->history);
-
- if (*index >= children)
- {
- *index = *index - children;
-
- if (*index < ephy_node_get_n_children (model->priv->bookmarks))
- {
- return model->priv->bookmarks;
- }
- else
- {
- return NULL;
- }
- }
- else
- {
- return model->priv->history;
- }
+free_search_terms (GSList *search_terms)
+{
+ GSList *iter;
+
+ for (iter = search_terms; iter != NULL; iter = iter->next)
+ g_regex_unref ((GRegex*)iter->data);
+
+ g_slist_free (search_terms);
}
static void
-root_child_removed_cb (EphyNode *node,
- EphyNode *child,
- guint old_index,
- EphyCompletionModel *model)
+ephy_completion_model_finalize (GObject *object)
{
- GtkTreePath *path;
- guint index;
-
- path = gtk_tree_path_new ();
+ EphyCompletionModelPrivate *priv = EPHY_COMPLETION_MODEL (object)->priv;
- index = old_index;
- if (node == model->priv->bookmarks)
- {
- index += ephy_node_get_n_children (model->priv->history);
- }
- gtk_tree_path_append_index (path, index);
+ if (priv->search_terms) {
+ free_search_terms (priv->search_terms);
+ priv->search_terms = NULL;
+ }
- gtk_tree_model_row_deleted (GTK_TREE_MODEL (model), path);
- gtk_tree_path_free (path);
+ G_OBJECT_CLASS (ephy_completion_model_parent_class)->finalize (object);
}
static void
-root_child_added_cb (EphyNode *node,
- EphyNode *child,
- EphyCompletionModel *model)
+ephy_completion_model_class_init (EphyCompletionModelClass *klass)
{
- GtkTreePath *path;
- GtkTreeIter iter;
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
- node_iter_from_node (model, node, child, &iter);
+ object_class->constructed = ephy_completion_model_constructed;
+ object_class->finalize = ephy_completion_model_finalize;
- path = get_path_real (model, node, child);
- gtk_tree_model_row_inserted (GTK_TREE_MODEL (model), path, &iter);
- gtk_tree_path_free (path);
+ g_type_class_add_private (object_class, sizeof (EphyCompletionModelPrivate));
}
static void
-root_child_changed_cb (EphyNode *node,
- EphyNode *child,
- guint property_id,
- EphyCompletionModel *model)
+ephy_completion_model_init (EphyCompletionModel *model)
{
- GtkTreePath *path;
- GtkTreeIter iter;
+ EphyCompletionModelPrivate *priv;
+ EphyBookmarks *bookmarks_service;
- node_iter_from_node (model, node, child, &iter);
+ model->priv = priv = EPHY_COMPLETION_MODEL_GET_PRIVATE (model);
- path = get_path_real (model, node, child);
- gtk_tree_model_row_changed (GTK_TREE_MODEL (model), path, &iter);
- gtk_tree_path_free (path);
-}
+ priv->browse_history = EPHY_BROWSE_HISTORY (ephy_embed_shell_get_global_browse_history (embed_shell));
+ priv->legacy_history_service = EPHY_HISTORY (ephy_embed_shell_get_global_history (embed_shell));
+ priv->favicon_cache = EPHY_FAVICON_CACHE (ephy_embed_shell_get_favicon_cache (embed_shell));
-static void
-connect_signals (EphyCompletionModel *model, EphyNode *root)
-{
- ephy_node_signal_connect_object (root,
- EPHY_NODE_CHILD_ADDED,
- (EphyNodeCallback)root_child_added_cb,
- G_OBJECT (model));
- ephy_node_signal_connect_object (root,
- EPHY_NODE_CHILD_REMOVED,
- (EphyNodeCallback)root_child_removed_cb,
- G_OBJECT (model));
- ephy_node_signal_connect_object (root,
- EPHY_NODE_CHILD_CHANGED,
- (EphyNodeCallback)root_child_changed_cb,
- G_OBJECT (model));
-}
-
-static void
-ephy_completion_model_init (EphyCompletionModel *model)
-{
- model->priv = EPHY_COMPLETION_MODEL_GET_PRIVATE (model);
- model->priv->stamp = g_random_int ();
-
- model->priv->history_service = EPHY_HISTORY (
- ephy_embed_shell_get_global_history (
- embed_shell));
- model->priv->history = ephy_history_get_pages (
- model->priv->history_service);
- connect_signals (model, model->priv->history);
-
- model->priv->bookmarks_service = ephy_shell_get_bookmarks (ephy_shell);
- model->priv->bookmarks = ephy_bookmarks_get_bookmarks (
- model->priv->bookmarks_service);
- connect_signals (model, model->priv->bookmarks);
-
- model->priv->favicon_cache = EPHY_FAVICON_CACHE (
- ephy_embed_shell_get_favicon_cache (
- EPHY_EMBED_SHELL (ephy_shell)));
+ bookmarks_service = ephy_shell_get_bookmarks (ephy_shell);
+ priv->bookmarks = ephy_bookmarks_get_bookmarks (bookmarks_service);
}
-EphyCompletionModel *
-ephy_completion_model_new (void)
+static gboolean
+is_base_address (const char *address)
{
- EphyCompletionModel *model;
-
- model = EPHY_COMPLETION_MODEL (g_object_new (EPHY_TYPE_COMPLETION_MODEL,
- NULL));
-
- g_return_val_if_fail (model->priv != NULL, NULL);
-
- return model;
+ if (address == NULL)
+ return FALSE;
+
+ /* A base address is <scheme>://<host>/
+ * Neither scheme nor host contain a slash, so we can use slashes
+ * figure out if it's a base address.
+ *
+ * Note: previous code was using a GRegExp to do the same thing.
+ * While regexps are much nicer to read, they're also a lot
+ * slower.
+ */
+ address = strchr (address, '/');
+ if (address == NULL ||
+ address[1] != '/')
+ return FALSE;
+
+ address += 2;
+ address = strchr (address, '/');
+ if (address == NULL ||
+ address[1] != 0)
+ return FALSE;
+
+ return TRUE;
}
static int
-ephy_completion_model_get_n_columns (GtkTreeModel *tree_model)
-{
- return N_COL;
+get_relevance (const char *location,
+ int visit_count,
+ gboolean is_bookmark)
+{
+ /* FIXME: use frecency. */
+ int relevance = 0;
+
+ /* We have three ordered groups: history's base addresses,
+ bookmarks, deep history addresses. */
+ if (is_bookmark)
+ relevance = 1 << 5;
+ else {
+ visit_count = MIN (visit_count, (1 << 5) - 1);
+
+ if (is_base_address (location))
+ relevance = visit_count << 10;
+ else
+ relevance = visit_count;
+ }
+
+ return relevance;
}
-static GType
-ephy_completion_model_get_column_type (GtkTreeModel *tree_model,
- int index)
-{
- GType type = 0;
-
- switch (index)
- {
- case EPHY_COMPLETION_TEXT_COL:
- case EPHY_COMPLETION_ACTION_COL:
- case EPHY_COMPLETION_KEYWORDS_COL:
- type = G_TYPE_STRING;
- break;
- case EPHY_COMPLETION_EXTRA_COL:
- type = G_TYPE_BOOLEAN;
- break;
- case EPHY_COMPLETION_FAVICON_COL:
- type = GDK_TYPE_PIXBUF;
- break;
- case EPHY_COMPLETION_RELEVANCE_COL:
- type = G_TYPE_INT;
- break;
- }
-
- return type;
-}
+typedef struct {
+ char *title;
+ char *location;
+ char *keywords;
+ int relevance;
+ gboolean is_bookmark;
+} PotentialRow;
static void
-init_text_col (GValue *value, EphyNode *node, int group)
-{
- const char *text;
-
- switch (group)
- {
- case BOOKMARKS_GROUP:
- case HISTORY_GROUP:
- text = ephy_node_get_property_string
- (node, EPHY_NODE_PAGE_PROP_TITLE);
- break;
-
- default:
- text = "";
- }
-
- g_value_set_string (value, text);
+set_row_in_model (EphyCompletionModel *model, int position, PotentialRow *row)
+{
+ const char *favicon_location = ephy_history_get_icon (model->priv->legacy_history_service,
+ row->location);
+
+ gtk_list_store_insert_with_values (GTK_LIST_STORE (model), NULL, position,
+ EPHY_COMPLETION_TEXT_COL, row->title ? row->title : "",
+ EPHY_COMPLETION_URL_COL, row->location,
+ EPHY_COMPLETION_ACTION_COL, row->location,
+ EPHY_COMPLETION_KEYWORDS_COL, row->keywords ? row->keywords : "",
+ EPHY_COMPLETION_EXTRA_COL, row->is_bookmark,
+ EPHY_COMPLETION_FAVICON_COL, ephy_favicon_cache_get (model->priv->favicon_cache,
+ favicon_location),
+ EPHY_COMPLETION_RELEVANCE_COL, row->relevance,
+ -1);
}
static void
-init_action_col (GValue *value, EphyNode *node)
+replace_rows_in_model (EphyCompletionModel *model, GSList *new_rows)
{
- const char *text;
+ /* This is by far the simplest way of doing, and yet it gives
+ * basically the same result than the other methods... */
+ int i;
- text = ephy_node_get_property_string
- (node, EPHY_NODE_BMK_PROP_LOCATION);
-
- g_value_set_string (value, text);
-}
+ gtk_list_store_clear (GTK_LIST_STORE (model));
-static void
-init_keywords_col (GValue *value, EphyNode *node, int group)
-{
- const char *text = NULL;
-
- switch (group)
- {
- case BOOKMARKS_GROUP:
- text = ephy_node_get_property_string
- (node, EPHY_NODE_BMK_PROP_KEYWORDS);
- break;
- }
-
- if (text == NULL)
- {
- text = "";
- }
-
- g_value_set_string (value, text);
-}
-static void
-init_favicon_col (EphyCompletionModel *model, GValue *value,
- EphyNode *node, int group)
-{
- const char *icon_location;
- GdkPixbuf *pixbuf = NULL;
- const char *url;
-
- switch (group)
- {
- case BOOKMARKS_GROUP:
- icon_location = ephy_node_get_property_string
- (node, EPHY_NODE_BMK_PROP_ICON);
- break;
- case HISTORY_GROUP:
- url = ephy_node_get_property_string
- (node, EPHY_NODE_PAGE_PROP_LOCATION);
- icon_location = ephy_history_get_icon (
- model->priv->history_service, url);
- break;
- default:
- icon_location = NULL;
- }
-
- if (icon_location)
- {
- pixbuf = ephy_favicon_cache_get (
- model->priv->favicon_cache, icon_location);
- }
-
- g_value_take_object (value, pixbuf);
+ for (i = 0; new_rows != NULL; i++) {
+ PotentialRow *row = (PotentialRow*)new_rows->data;
+
+ set_row_in_model (model, i, row);
+ new_rows = new_rows->next;
+ }
}
static gboolean
-is_base_address (const char *address)
-{
- if (address == NULL)
- return FALSE;
-
- /* a base address is <scheme>://<host>/
- * Neither scheme nor host contain a slash, so we can use slashes
- * figure out if it's a base address.
- *
- * Note: previous code was using a GRegExp to do the same thing.
- * While regexps are much nicer to read, they're also a lot
- * slower.
- */
- address = strchr (address, '/');
- if (address == NULL ||
- address[1] != '/')
- return FALSE;
-
- address += 2;
- address = strchr (address, '/');
- if (address == NULL ||
- address[1] != 0)
- return FALSE;
-
- return TRUE;
-}
-
-static void
-init_relevance_col (GValue *value, EphyNode *node, int group)
-{
- int relevance = 0;
-
- /* We have three ordered groups: history's base
- addresses, bookmarks, deep history addresses */
-
- if (group == BOOKMARKS_GROUP)
- {
- relevance = 1 << 5;
- }
- else if (group == HISTORY_GROUP)
- {
- const char *address;
- int visits;
-
- visits = ephy_node_get_property_int
- (node, EPHY_NODE_PAGE_PROP_VISITS);
- address = ephy_node_get_property_string
- (node, EPHY_NODE_PAGE_PROP_LOCATION);
-
- visits = MIN (visits, (1 << 5) - 1);
-
- if (is_base_address (address))
- {
- relevance = visits << 10;
- }
- else
- {
- relevance = visits;
- }
- }
-
- g_value_set_int (value, relevance);
-}
-
-static void
-init_url_col (GValue *value, EphyNode *node)
-{
- const char *url = NULL;
-
- url = ephy_node_get_property_string
- (node, EPHY_NODE_PAGE_PROP_LOCATION);
-
- g_value_set_string (value, url);
-}
-
-static void
-ephy_completion_model_get_value (GtkTreeModel *tree_model,
- GtkTreeIter *iter,
- int column,
- GValue *value)
-{
- int group;
- EphyCompletionModel *model = EPHY_COMPLETION_MODEL (tree_model);
- EphyNode *node;
-
- g_return_if_fail (EPHY_IS_COMPLETION_MODEL (tree_model));
- g_return_if_fail (iter != NULL);
- g_return_if_fail (iter->stamp == model->priv->stamp);
-
- node = iter->user_data;
- group = (iter->user_data2 == model->priv->history) ?
- HISTORY_GROUP : BOOKMARKS_GROUP;
-
- switch (column)
- {
- case EPHY_COMPLETION_EXTRA_COL:
- g_value_init (value, G_TYPE_BOOLEAN);
- g_value_set_boolean (value, (group == BOOKMARKS_GROUP));
- break;
- case EPHY_COMPLETION_TEXT_COL:
- g_value_init (value, G_TYPE_STRING);
- init_text_col (value, node, group);
- break;
- case EPHY_COMPLETION_FAVICON_COL:
- g_value_init (value, GDK_TYPE_PIXBUF);
- init_favicon_col (model, value, node, group);
- break;
- case EPHY_COMPLETION_ACTION_COL:
- g_value_init (value, G_TYPE_STRING);
- init_action_col (value, node);
- break;
- case EPHY_COMPLETION_KEYWORDS_COL:
- g_value_init (value, G_TYPE_STRING);
- init_keywords_col (value, node, group);
- break;
- case EPHY_COMPLETION_RELEVANCE_COL:
- g_value_init (value, G_TYPE_INT);
- init_relevance_col (value, node, group);
- break;
- case EPHY_COMPLETION_URL_COL:
- g_value_init (value, G_TYPE_STRING);
- init_url_col (value, node);
- break;
- }
-}
-
-static GtkTreeModelFlags
-ephy_completion_model_get_flags (GtkTreeModel *tree_model)
-{
- return GTK_TREE_MODEL_ITERS_PERSIST | GTK_TREE_MODEL_LIST_ONLY;
+should_add_bookmark_to_model (EphyCompletionModel *model,
+ const char *search_string,
+ const char *title,
+ const char *location,
+ const char *keywords)
+{
+ gboolean ret = TRUE;
+ EphyCompletionModelPrivate *priv = model->priv;
+
+ if (priv->search_terms) {
+ GSList *iter;
+ GRegex *current = NULL;
+
+ for (iter = priv->search_terms; iter != NULL; iter = iter->next) {
+ current = (GRegex*) iter->data;
+ if ((!g_regex_match (current, title, G_REGEX_MATCH_NOTEMPTY, NULL)) &&
+ (!g_regex_match (current, location, G_REGEX_MATCH_NOTEMPTY, NULL)) &&
+ (!g_regex_match (current, keywords, G_REGEX_MATCH_NOTEMPTY, NULL))) {
+ ret = FALSE;
+ break;
+ }
+ }
+ }
+
+ return ret;
}
-static gboolean
-ephy_completion_model_get_iter (GtkTreeModel *tree_model,
- GtkTreeIter *iter,
- GtkTreePath *path)
-{
- EphyCompletionModel *model = EPHY_COMPLETION_MODEL (tree_model);
- EphyNode *root, *child;
- int i;
-
- g_return_val_if_fail (EPHY_IS_COMPLETION_MODEL (model), FALSE);
- g_return_val_if_fail (gtk_tree_path_get_depth (path) > 0, FALSE);
-
- i = gtk_tree_path_get_indices (path)[0];
-
- root = get_index_root (model, &i);
- if (root == NULL) return FALSE;
+typedef struct {
+ EphyCompletionModel *model;
+ char *search_string;
+ EphyHistoryJobCallback callback;
+ gpointer user_data;
+} FindURLsData;
- child = ephy_node_get_nth_child (root, i);
- g_return_val_if_fail (child != NULL, FALSE);
-
- node_iter_from_node (model, root, child, iter);
-
- return TRUE;
-}
-
-static GtkTreePath *
-ephy_completion_model_get_path (GtkTreeModel *tree_model,
- GtkTreeIter *iter)
+static int
+find_url (gconstpointer a,
+ gconstpointer b)
{
- EphyCompletionModel *model = EPHY_COMPLETION_MODEL (tree_model);
-
- g_return_val_if_fail (iter != NULL, NULL);
- g_return_val_if_fail (iter->user_data != NULL, NULL);
- g_return_val_if_fail (iter->user_data2 != NULL, NULL);
- g_return_val_if_fail (iter->stamp == model->priv->stamp, NULL);
-
- return get_path_real (model, iter->user_data2, iter->user_data);
+ return g_strcmp0 (((PotentialRow*)a)->location,
+ ((char *)b));
}
-static gboolean
-ephy_completion_model_iter_next (GtkTreeModel *tree_model,
- GtkTreeIter *iter)
+static PotentialRow *
+potential_row_new (const char *title, const char *location,
+ const char *keywords, int visit_count,
+ gboolean is_bookmark)
{
- EphyCompletionModel *model = EPHY_COMPLETION_MODEL (tree_model);
- EphyNode *node, *next, *root;
-
- g_return_val_if_fail (iter != NULL, FALSE);
- g_return_val_if_fail (iter->user_data != NULL, FALSE);
- g_return_val_if_fail (iter->user_data2 != NULL, FALSE);
- g_return_val_if_fail (iter->stamp == model->priv->stamp, FALSE);
-
- node = iter->user_data;
- root = iter->user_data2;
+ PotentialRow *row = g_slice_new0 (PotentialRow);
- next = ephy_node_get_next_child (root, node);
+ row->title = g_strdup (title);
+ row->location = g_strdup (location);
+ row->keywords = g_strdup (keywords);
+ row->relevance = get_relevance (location, visit_count, is_bookmark);
+ row->is_bookmark = is_bookmark;
- if (next == NULL && root == model->priv->history)
- {
- root = model->priv->bookmarks;
- next = ephy_node_get_nth_child (model->priv->bookmarks, 0);
- }
-
- if (next == NULL) return FALSE;
-
- node_iter_from_node (model, root, next, iter);
-
- return TRUE;
+ return row;
}
-static gboolean
-ephy_completion_model_iter_children (GtkTreeModel *tree_model,
- GtkTreeIter *iter,
- GtkTreeIter *parent)
+static void
+free_potential_row (PotentialRow *row)
{
- EphyCompletionModel *model = EPHY_COMPLETION_MODEL (tree_model);
- EphyNode *root, *first_node;
-
- if (parent != NULL)
- {
- return FALSE;
- }
-
- root = model->priv->history;
- first_node = ephy_node_get_nth_child (root, 0);
-
- if (first_node == NULL)
- {
- root = model->priv->bookmarks;
- first_node = ephy_node_get_nth_child (root, 0);
- }
-
- if (first_node == NULL)
- {
- return FALSE;
- }
-
- node_iter_from_node (model, root, first_node, iter);
-
- return TRUE;
+ g_free (row->title);
+ g_free (row->location);
+ g_free (row->keywords);
+
+ g_slice_free (PotentialRow, row);
}
-static gboolean
-ephy_completion_model_iter_has_child (GtkTreeModel *tree_model,
- GtkTreeIter *iter)
-{
- return FALSE;
+static GSList *
+add_to_potential_rows (GSList *rows,
+ const char *title,
+ const char *location,
+ const char *keywords,
+ int visit_count,
+ gboolean is_bookmark,
+ gboolean search_for_duplicates)
+{
+ gboolean found = FALSE;
+ PotentialRow *row = potential_row_new (title, location, keywords, visit_count, is_bookmark);
+
+ if (search_for_duplicates) {
+ GSList *p;
+
+ p = g_slist_find_custom (rows, location, find_url);
+ if (p) {
+ PotentialRow *match = (PotentialRow*)p->data;
+ if (row->relevance > match->relevance)
+ match->relevance = row->relevance;
+
+ found = TRUE;
+ free_potential_row (row);
+ }
+ }
+
+ if (!found)
+ rows = g_slist_prepend (rows, row);
+
+ return rows;
}
static int
-ephy_completion_model_iter_n_children (GtkTreeModel *tree_model,
- GtkTreeIter *iter)
+sort_by_relevance (gconstpointer a, gconstpointer b)
{
- EphyCompletionModel *model = EPHY_COMPLETION_MODEL (tree_model);
-
- g_return_val_if_fail (EPHY_IS_COMPLETION_MODEL (tree_model), -1);
-
- if (iter == NULL)
- {
- return ephy_node_get_n_children (model->priv->history) +
- ephy_node_get_n_children (model->priv->bookmarks);
- }
+ PotentialRow *r1 = (PotentialRow*)a;
+ PotentialRow *r2 = (PotentialRow*)b;
- g_return_val_if_fail (model->priv->stamp == iter->stamp, -1);
-
- return 0;
+ if (r1->relevance < r2->relevance)
+ return 1;
+ else if (r1->relevance > r2->relevance)
+ return -1;
+ else
+ return 0;
}
-static gboolean
-ephy_completion_model_iter_nth_child (GtkTreeModel *tree_model,
- GtkTreeIter *iter,
- GtkTreeIter *parent,
- int n)
-{
- EphyCompletionModel *model = EPHY_COMPLETION_MODEL (tree_model);
- EphyNode *node, *root;
-
- g_return_val_if_fail (EPHY_IS_COMPLETION_MODEL (tree_model), FALSE);
-
- if (parent != NULL)
- {
- return FALSE;
- }
-
- root = get_index_root (model, &n);
- node = ephy_node_get_nth_child (root, n);
-
- if (node == NULL) return FALSE;
-
- node_iter_from_node (model, root, node, iter);
+static void
+query_completed_cb (EphyHistoryService *service,
+ gboolean success,
+ gpointer result_data,
+ FindURLsData *user_data)
+{
+ EphyCompletionModel *model = user_data->model;
+ EphyCompletionModelPrivate *priv = model->priv;
+ GList *p, *urls;
+ GPtrArray *children;
+ GSList *list = NULL;
+ int i;
+
+ /* Bookmarks */
+ children = ephy_node_get_children (priv->bookmarks);
+
+ /* FIXME: perhaps this could be done in a service thread? There
+ * should never be a ton of bookmarks, but seems a bit cleaner and
+ * consistent with what we do for the history. */
+ for (i = 0; i < children->len; i++) {
+ EphyNode *kid;
+ const char *keywords, *location, *title;
+
+ kid = g_ptr_array_index (children, i);
+ location = ephy_node_get_property_string (kid, EPHY_NODE_BMK_PROP_LOCATION);
+ title = ephy_node_get_property_string (kid, EPHY_NODE_BMK_PROP_TITLE);
+ keywords = ephy_node_get_property_string (kid, EPHY_NODE_BMK_PROP_KEYWORDS);
+
+ if (should_add_bookmark_to_model (model, user_data->search_string,
+ title, location, keywords))
+ list = add_to_potential_rows (list, title, location, keywords, 0, TRUE, FALSE);
+ }
+
+ /* History */
+ urls = (GList*)result_data;
+
+ for (p = urls; p != NULL; p = p->next) {
+ EphyHistoryURL *url = (EphyHistoryURL*)p->data;
+
+ list = add_to_potential_rows (list, url->title, url->url, NULL, url->visit_count, FALSE, TRUE);
+ }
+
+ /* Sort the rows by relevance. */
+ list = g_slist_sort (list, sort_by_relevance);
+
+ /* Now that we have all the rows we want to insert, replace the rows
+ * in the current model one by one, sorted by relevance. */
+ replace_rows_in_model (model, list);
+
+ /* Notify */
+ if (user_data->callback)
+ user_data->callback (service, success, result_data, user_data->user_data);
+
+ g_free (user_data->search_string);
+ g_slice_free (FindURLsData, user_data);
+
+ g_slist_free_full (list, (GDestroyNotify)free_potential_row);
+}
- return TRUE;
+static void
+update_search_terms (EphyCompletionModel *model,
+ const char *text)
+{
+ const char *current;
+ const char *ptr;
+ char *tmp;
+ char *term;
+ GRegex *term_regex;
+ GRegex *quote_regex;
+ gint count;
+ gboolean inside_quotes = FALSE;
+ EphyCompletionModelPrivate *priv = model->priv;
+
+ if (priv->search_terms) {
+ free_search_terms (priv->search_terms);
+ priv->search_terms = NULL;
+ }
+
+ quote_regex = g_regex_new ("\"", G_REGEX_OPTIMIZE,
+ G_REGEX_MATCH_NOTEMPTY, NULL);
+
+ /*
+ * This code loops through the string using pointer arythmetics.
+ * Although the string we are handling may contain UTF-8 chars
+ * this works because only ASCII chars affect what is actually
+ * copied from the string as a search term.
+ */
+ for (count = 0, current = ptr = text; ptr[0] != '\0'; ptr++, count++) {
+ /*
+ * If we found a double quote character; we will
+ * consume bytes up until the next quote, or
+ * end of line;
+ */
+ if (ptr[0] == '"')
+ inside_quotes = !inside_quotes;
+
+ /*
+ * If we found a space, and we are not looking for a
+ * closing double quote, or if the next char is the
+ * end of the string, append what we have already as
+ * a search term.
+ */
+ if (((ptr[0] == ' ') && (!inside_quotes)) || ptr[1] == '\0') {
+ /*
+ * We special-case the end of the line because
+ * we would otherwise not copy the last character
+ * of the search string, since the for loop will
+ * stop before that.
+ */
+ if (ptr[1] == '\0')
+ count++;
+
+ /*
+ * remove quotes, and quote any regex-sensitive
+ * characters
+ */
+ tmp = g_regex_escape_string (current, count);
+ term = g_regex_replace (quote_regex, tmp, -1, 0,
+ "", G_REGEX_MATCH_NOTEMPTY, NULL);
+ g_strstrip (term);
+ g_free (tmp);
+
+ /* we don't want empty search terms */
+ if (term[0] != '\0') {
+ term_regex = g_regex_new (term,
+ G_REGEX_CASELESS | G_REGEX_OPTIMIZE,
+ G_REGEX_MATCH_NOTEMPTY, NULL);
+ priv->search_terms = g_slist_append (priv->search_terms, term_regex);
+ }
+ g_free (term);
+
+ /* count will be incremented by the for loop */
+ count = -1;
+ current = ptr + 1;
+ }
+ }
+
+ g_regex_unref (quote_regex);
}
-static gboolean
-ephy_completion_model_iter_parent (GtkTreeModel *tree_model,
- GtkTreeIter *iter,
- GtkTreeIter *child)
-{
- return FALSE;
+#define MAX_COMPLETION_HISTORY_URLS 8
+
+void
+ephy_completion_model_update_for_string (EphyCompletionModel *model,
+ const char *search_string,
+ EphyHistoryJobCallback callback,
+ gpointer data)
+{
+ EphyCompletionModelPrivate *priv;
+ char **strings;
+ int i;
+ GList *query = NULL;
+ FindURLsData *user_data;
+
+ g_return_if_fail (EPHY_IS_COMPLETION_MODEL (model));
+ g_return_if_fail (search_string != NULL);
+
+ priv = model->priv;
+
+ /* Split the search string. */
+ strings = g_strsplit (search_string, " ", -1);
+ for (i = 0; strings[i]; i++)
+ query = g_list_append (query, g_strdup (strings[i]));
+ g_strfreev (strings);
+
+ update_search_terms (model, search_string);
+
+ user_data = g_slice_new (FindURLsData);
+ user_data->model = model;
+ user_data->search_string = g_strdup (search_string);
+ user_data->callback = callback;
+ user_data->user_data = data;
+
+ ephy_browse_history_find_urls (priv->browse_history,
+ 0, 0,
+ MAX_COMPLETION_HISTORY_URLS,
+ query,
+ (EphyHistoryJobCallback)query_completed_cb,
+ user_data);
}
-static void
-ephy_completion_model_tree_model_init (GtkTreeModelIface *iface)
+EphyCompletionModel *
+ephy_completion_model_new (void)
{
- iface->get_flags = ephy_completion_model_get_flags;
- iface->get_iter = ephy_completion_model_get_iter;
- iface->get_path = ephy_completion_model_get_path;
- iface->iter_next = ephy_completion_model_iter_next;
- iface->iter_children = ephy_completion_model_iter_children;
- iface->iter_has_child = ephy_completion_model_iter_has_child;
- iface->iter_n_children = ephy_completion_model_iter_n_children;
- iface->iter_nth_child = ephy_completion_model_iter_nth_child;
- iface->iter_parent = ephy_completion_model_iter_parent;
- iface->get_n_columns = ephy_completion_model_get_n_columns;
- iface->get_column_type = ephy_completion_model_get_column_type;
- iface->get_value = ephy_completion_model_get_value;
+ return g_object_new (EPHY_TYPE_COMPLETION_MODEL, NULL);
}
diff --git a/src/ephy-completion-model.h b/src/ephy-completion-model.h
index 62cae95c6..f02f81242 100644
--- a/src/ephy-completion-model.h
+++ b/src/ephy-completion-model.h
@@ -1,5 +1,5 @@
/*
- * Copyright © 2003 Marco Pesenti Gritti <marco@gnome.org>
+ * Copyright © 2012 Igalia S.L.
*
* 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
@@ -24,6 +24,8 @@
#ifndef EPHY_COMPLETION_MODEL_H
#define EPHY_COMPLETION_MODEL_H
+#include "ephy-history-service.h"
+
#include <gtk/gtk.h>
G_BEGIN_DECLS
@@ -51,7 +53,7 @@ typedef enum
typedef struct
{
- GObject parent;
+ GtkListStore parent;
/*< private >*/
EphyCompletionModelPrivate *priv;
@@ -59,13 +61,17 @@ typedef struct
typedef struct
{
- GObjectClass parent;
+ GtkListStoreClass parent;
} EphyCompletionModelClass;
-GType ephy_completion_model_get_type (void);
+GType ephy_completion_model_get_type (void);
-EphyCompletionModel *ephy_completion_model_new (void);
+EphyCompletionModel *ephy_completion_model_new (void);
+void ephy_completion_model_update_for_string (EphyCompletionModel *model,
+ const char *string,
+ EphyHistoryJobCallback callback,
+ gpointer data);
G_END_DECLS
#endif /* EPHY_COMPLETION_MODEL_H */
diff --git a/src/ephy-location-controller.c b/src/ephy-location-controller.c
index b0509ddaf..9315ad6cf 100644
--- a/src/ephy-location-controller.c
+++ b/src/ephy-location-controller.c
@@ -96,51 +96,8 @@ match_func (GtkEntryCompletion *completion,
GtkTreeIter *iter,
gpointer data)
{
- char *item = NULL;
- char *url = NULL;
- char *keywords = NULL;
-
- gboolean ret = FALSE;
- GtkTreeModel *model;
- GSList *search_terms;
-
- model = gtk_entry_completion_get_model (completion);
-
- gtk_tree_model_get (model, iter,
- EPHY_COMPLETION_TEXT_COL, &item,
- EPHY_COMPLETION_URL_COL, &url,
- EPHY_COMPLETION_KEYWORDS_COL, &keywords,
- -1);
-
- if (!key)
- return FALSE;
-
- search_terms = ephy_location_entry_get_search_terms (data);
-
- if (search_terms)
- {
- GSList *iter;
- GRegex *current = NULL;
-
- ret = TRUE;
- for (iter = search_terms; iter != NULL; iter = iter->next)
- {
- current = (GRegex*) iter->data;
- if ((!g_regex_match (current, item, G_REGEX_MATCH_NOTEMPTY, NULL)) &&
- (!g_regex_match (current, url, G_REGEX_MATCH_NOTEMPTY, NULL)) &&
- (!g_regex_match (current, keywords, G_REGEX_MATCH_NOTEMPTY, NULL)))
- {
- ret = FALSE;
- break;
- }
- }
- }
-
- g_free (item);
- g_free (url);
- g_free (keywords);
-
- return ret;
+ /* We want every row in the model to show up. */
+ return TRUE;
}
static void
@@ -211,6 +168,8 @@ static void
user_changed_cb (GtkWidget *widget, EphyLocationController *controller)
{
const char *address;
+ GtkTreeModel *model;
+ GtkEntryCompletion *completion;
address = ephy_location_entry_get_location (EPHY_LOCATION_ENTRY (widget));
@@ -218,6 +177,12 @@ user_changed_cb (GtkWidget *widget, EphyLocationController *controller)
g_signal_handlers_block_by_func (controller, G_CALLBACK (sync_address), widget);
ephy_location_controller_set_address (controller, address);
+
+ completion = gtk_entry_get_completion (GTK_ENTRY (widget));
+ model = gtk_entry_completion_get_model (completion);
+
+ ephy_completion_model_update_for_string (EPHY_COMPLETION_MODEL (model), address,
+ NULL, NULL);
g_signal_handlers_unblock_by_func (controller, G_CALLBACK (sync_address), widget);
}
@@ -428,7 +393,7 @@ ephy_location_controller_constructed (GObject *object)
EPHY_COMPLETION_EXTRA_COL,
EPHY_COMPLETION_FAVICON_COL);
g_object_unref (model);
-
+
ephy_location_entry_set_match_func (priv->location_entry,
match_func,
priv->location_entry,