diff options
Diffstat (limited to 'src/bookmarks/ephy-topics-entry.c')
-rw-r--r-- | src/bookmarks/ephy-topics-entry.c | 470 |
1 files changed, 290 insertions, 180 deletions
diff --git a/src/bookmarks/ephy-topics-entry.c b/src/bookmarks/ephy-topics-entry.c index 64145ca66..6153e53a7 100644 --- a/src/bookmarks/ephy-topics-entry.c +++ b/src/bookmarks/ephy-topics-entry.c @@ -41,8 +41,8 @@ struct _EphyTopicsEntryPrivate EphyNode *bookmark; GtkListStore *store; GtkEntryCompletion *completion; - gboolean update_keywords; - char *input; + gboolean lock; + char *create; char *key; }; @@ -53,6 +53,14 @@ enum PROP_BOOKMARK }; +enum +{ + COLUMN_NODE, + COLUMN_KEY, + COLUMN_TITLE, + COLUMNS +}; + static GObjectClass *parent_class = NULL; GType @@ -83,6 +91,76 @@ ephy_topics_entry_get_type (void) return type; } +static EphyNode * +find_topic (EphyTopicsEntry *entry, + const char *key) +{ + EphyNode *node = NULL; + GtkTreeModel *model; + GtkTreeIter iter; + GValue value = { 0, }; + gboolean valid; + + /* Loop through our table and set/unset topics appropriately */ + model = GTK_TREE_MODEL (entry->priv->store); + valid = gtk_tree_model_get_iter_first (model, &iter); + while (valid && node == NULL) + { + gtk_tree_model_get_value (model, &iter, COLUMN_KEY, &value); + if (strcmp(g_value_get_string (&value), key) == 0) + { + g_value_unset (&value); + gtk_tree_model_get_value (model, &iter, COLUMN_NODE, &value); + node = g_value_get_pointer (&value); + } + g_value_unset (&value); + valid = gtk_tree_model_iter_next (model, &iter); + } + + return node; +} + +static void +insert_text (EphyTopicsEntry *entry, + const char *title) +{ + GtkEditable *editable = GTK_EDITABLE (entry); + const char *text = gtk_entry_get_text (GTK_ENTRY (entry)); + gint start, end; + + /* Find the start and end locations */ + start = gtk_editable_get_position (editable); + while (start > 0 && text[start-1] != ';') + { + start--; + } + if(start > 0 && text[start-1] == ';' && text[start] == ' ') + { + start++; + } + end = start; + while (text[end] && text[end] != ';') + { + end++; + } + end = end; + if (text[end] == ';') + { + end++; + if (text[end] == ' ') + { + end++; + } + } + + /* Replace the text in the current position with the title */ + gtk_editable_delete_text (editable, start, end); + gtk_editable_insert_text (editable, title, strlen(title), &start); + gtk_editable_insert_text (editable, "; ", 2, &start); + gtk_editable_set_position (editable, start); +} + +/* Updates the text entry and the completion model to match the database */ static void update_widget (EphyTopicsEntry *entry) { @@ -97,7 +175,9 @@ update_widget (EphyTopicsEntry *entry) char *tmp1, *tmp2; gboolean update_text; - priv->update_keywords = FALSE; + /* Prevent any changes to the database */ + if(priv->lock) return; + priv->lock = TRUE; node = ephy_bookmarks_get_keywords (priv->bookmarks); children = ephy_node_get_children (node); @@ -107,7 +187,8 @@ update_widget (EphyTopicsEntry *entry) { node = g_ptr_array_index (children, i); - priority = ephy_node_get_property_int (node, EPHY_NODE_KEYWORD_PROP_PRIORITY); + priority = ephy_node_get_property_int + (node, EPHY_NODE_KEYWORD_PROP_PRIORITY); if (priority != EPHY_NODE_NORMAL_PRIORITY) continue; @@ -115,116 +196,65 @@ update_widget (EphyTopicsEntry *entry) } g_ptr_array_sort (topics, ephy_bookmarks_compare_topic_pointers); - gtk_list_store_clear (priv->store); + update_text = !GTK_WIDGET_HAS_FOCUS (GTK_WIDGET (entry)); - if (update_text) gtk_editable_delete_text (editable, 0, -1); - pos = 0; + if (update_text) + { + gtk_editable_delete_text (editable, 0, -1); + } - for (i = 0; i < topics->len; i++) + for (pos = 0, i = 0; i < topics->len; i++) { node = g_ptr_array_index (topics, i); title = ephy_node_get_property_string (node, EPHY_NODE_KEYWORD_PROP_NAME); - if (ephy_node_has_child (node, priv->bookmark)) + if (update_text && ephy_node_has_child (node, priv->bookmark)) { - if (update_text) - { - gtk_editable_insert_text (editable, title, -1, &pos); - gtk_editable_insert_text (editable, "; ", -1, &pos); - } + gtk_editable_insert_text (editable, title, -1, &pos); + gtk_editable_insert_text (editable, "; ", -1, &pos); } - /* We just include all topics, else we get odd behaviour after you - * just finish entering a topic (it gets selected, so wouldn't show) */ tmp1 = g_utf8_casefold (title, -1); tmp2 = g_utf8_normalize (tmp1, -1, G_NORMALIZE_DEFAULT); gtk_list_store_append (priv->store, &iter); - gtk_list_store_set (priv->store, &iter, 0, title, 1, tmp2, -1); + gtk_list_store_set (priv->store, &iter, COLUMN_NODE, node, + COLUMN_TITLE, title, COLUMN_KEY, tmp2, -1); g_free (tmp2); g_free (tmp1); } - g_ptr_array_free (topics, TRUE); - if (update_text) gtk_editable_set_position (editable, -1); - - priv->update_keywords = TRUE; -} - -static void -update_input (EphyTopicsEntry *entry) -{ - GtkEditable *editable = GTK_EDITABLE (entry); - const char *text = gtk_entry_get_text (GTK_ENTRY (entry)); - char *input; - gint start, end; - - if (entry->priv->input) + if (update_text) { - gtk_entry_completion_delete_action (entry->priv->completion, 0); - g_free (entry->priv->input); - g_free (entry->priv->key); - entry->priv->input = 0; - entry->priv->key = 0; + gtk_editable_set_position (editable, -1); } - /* Find the start and end locations */ - start = gtk_editable_get_position (editable); - while (start > 0 && text[start-1] != ';') - { - start--; - } - if(start > 0 && text[start-1] == ';' && text[start] == ' ') - { - start++; - } - end = start; - while (text[end] && text[end] != ';') - { - end++; - } - - /* If no text to work with, exit */ - input = g_strndup (text+start, end-start); - g_strstrip (input); - if (*input != 0) - { - entry->priv->input = input; - - input = g_utf8_casefold (input, -1); - entry->priv->key = g_utf8_normalize (input, -1, G_NORMALIZE_DEFAULT); - g_free (input); - - if (ephy_bookmarks_find_keyword (entry->priv->bookmarks, entry->priv->input, FALSE) == NULL) - { - input = g_strdup_printf (_("Create topic ā%sā"), entry->priv->input); - gtk_entry_completion_insert_action_text (entry->priv->completion, 0, input); - g_free (input); - } - else - { - g_free (entry->priv->input); - entry->priv->input = 0; - } - } - else - { - g_free (input); - } + g_ptr_array_free (topics, TRUE); + + priv->lock = FALSE; } +/* Updates the bookmarks database to match what is in the text entry */ static void -update_keywords (EphyTopicsEntry *entry) +update_database (EphyTopicsEntry *entry) { + EphyTopicsEntryPrivate *priv = entry->priv; + EphyNode *node; - GPtrArray *children; const char *text; char **split; - char *title, *tmp; - gint i, j, priority; + char *tmp; + gint i; - if(!entry->priv->update_keywords) return; + GtkTreeModel *model; + GtkTreeIter iter; + GValue value = { 0, }; + gboolean valid; + + /* Prevent any changes to the text entry or completion model */ + if(priv->lock) return; + priv->lock = TRUE; /* Get the list of strings input by the user */ text = gtk_entry_get_text (GTK_ENTRY (entry)); @@ -240,53 +270,63 @@ update_keywords (EphyTopicsEntry *entry) g_free (tmp); } - /* Test each keyword and set/unset as appropriate */ - node = ephy_bookmarks_get_keywords (entry->priv->bookmarks); - children = ephy_node_get_children (node); - for (i = 0; i < children->len; i++) + /* Loop through the completion model and set/unset topics appropriately */ + model = GTK_TREE_MODEL (priv->store); + valid = gtk_tree_model_get_iter_first (model, &iter); + while (valid) { - node = g_ptr_array_index (children, i); - - priority = ephy_node_get_property_int (node, EPHY_NODE_KEYWORD_PROP_PRIORITY); - if (priority != EPHY_NODE_NORMAL_PRIORITY) - continue; + gtk_tree_model_get_value (model, &iter, COLUMN_NODE, &value); + node = g_value_get_pointer (&value); + g_value_unset (&value); - text = ephy_node_get_property_string (node, EPHY_NODE_KEYWORD_PROP_NAME); - tmp = g_utf8_casefold (text, -1); - title = g_utf8_normalize (tmp, -1, G_NORMALIZE_DEFAULT); - g_free (tmp); + gtk_tree_model_get_value (model, &iter, COLUMN_KEY, &value); + text = g_value_get_string (&value); - for (j=0; split[j]; j++) - if (strcmp (title, split[j]) == 0) + for (i=0; split[i]; i++) + if (strcmp (text, split[i]) == 0) break; - if (split[j]) + if (split[i]) { - split[j][0] = 0; - ephy_bookmarks_set_keyword (entry->priv->bookmarks, node, - entry->priv->bookmark); + split[i][0] = 0; + ephy_bookmarks_set_keyword (priv->bookmarks, node, + priv->bookmark); } else { - ephy_bookmarks_unset_keyword (entry->priv->bookmarks, node, - entry->priv->bookmark); + ephy_bookmarks_unset_keyword (priv->bookmarks, node, + priv->bookmark); } - g_free (title); + g_value_unset (&value); + valid = gtk_tree_model_iter_next (model, &iter); } - - g_strfreev (split); + + priv->lock = FALSE; } +/* Updates the search key and topic creation action */ static void -insert_text (EphyTopicsEntry *entry, - const char *title) +update_key (EphyTopicsEntry *entry) { GtkEditable *editable = GTK_EDITABLE (entry); + EphyTopicsEntryPrivate *priv = entry->priv; + const char *text = gtk_entry_get_text (GTK_ENTRY (entry)); - char *key; + char *input; gint start, end; + /* If there was something we could create, then delete the action. */ + if (priv->create) + { + gtk_entry_completion_delete_action (priv->completion, 0); + } + + g_free (priv->create); + g_free (priv->key); + priv->create = 0; + priv->key = 0; + /* Find the start and end locations */ start = gtk_editable_get_position (editable); while (start > 0 && text[start-1] != ';') @@ -302,36 +342,94 @@ insert_text (EphyTopicsEntry *entry, { end++; } - end = end; - if (text[end] == ';') + + /* Set the priv->create and priv->key appropriately. */ + input = g_strndup (text+start, end-start); + g_strstrip (input); + if (*input != 0) { - end++; - if (text[end] == ' ') + priv->create = input; + + input = g_utf8_casefold (input, -1); + priv->key = g_utf8_normalize (input, -1, G_NORMALIZE_DEFAULT); + + if (find_topic (entry, priv->key) != NULL) { - end++; + g_free (priv->create); + priv->create = 0; } } + g_free (input); + + /* If there is something we can create, then setup the action. */ + if (priv->create) + { + input = g_strdup_printf (_("Create topic ā%sā"), priv->create); + gtk_entry_completion_insert_action_text (priv->completion, 0, input); + g_free (input); + } +} + +static gboolean +match_func (GtkEntryCompletion *completion, + const gchar *key, + GtkTreeIter *iter, + gpointer user_data) +{ + EphyTopicsEntry *entry = EPHY_TOPICS_ENTRY (gtk_entry_completion_get_entry (completion)); + EphyTopicsEntryPrivate *priv = entry->priv; + GtkTreeModel *model = gtk_entry_completion_get_model (completion); + + gboolean result; + GValue value = { 0, }; + EphyNode *node; - /* If we were provided with nothing, then we're meant to find a topic - * or create one from whatever has been entered */ - if (title == NULL) + /* If no node at all (this happens for unknown reasons) then don't show. */ + gtk_tree_model_get_value (model, iter, COLUMN_NODE, &value); + node = g_value_get_pointer (&value); + if (node == NULL) + { + g_value_unset (&value); + result = FALSE; + } + + /* If it's already selected, don't show it. */ + else if (ephy_node_has_child (node, priv->bookmark)) { - /* If no text to work with, exit */ - key = g_strndup (text+start, end-start); - g_strstrip (key); - if (*key == 0) + g_value_unset (&value); + if (priv->key == NULL) + { + result = FALSE; + } + + /* Unless it's the one we're currently editing. */ + else { - g_free (key); - return; + gtk_tree_model_get_value (model, iter, COLUMN_KEY, &value); + result = (strcmp (g_value_get_string (&value), priv->key) == 0); + g_value_unset (&value); } } - - /* Replace the text in the current position with the title */ - gtk_editable_delete_text (editable, start, end); - gtk_editable_insert_text (editable, title, strlen(title), &start); - gtk_editable_insert_text (editable, "; ", 2, &start); - gtk_editable_set_position (editable, start); - update_widget(entry); + + /* If it's not selected, assume we show it. */ + else + { + g_value_unset (&value); + if (priv->key == NULL) + { + result = TRUE; + } + + /* Unless it's not matching the current key. */ + else + { + gtk_tree_model_get_value (model, iter, COLUMN_KEY, &value); + result = (g_str_has_prefix (g_value_get_string (&value), priv->key)); + g_value_unset (&value); + } + } + + return result; } static void @@ -340,15 +438,16 @@ action_cb (GtkEntryCompletion *completion, gpointer user_data) { EphyTopicsEntry *entry = EPHY_TOPICS_ENTRY (gtk_entry_completion_get_entry (completion)); - char *action = g_strdup(entry->priv->input); - - if (ephy_bookmarks_find_keyword (entry->priv->bookmarks, action, FALSE) == NULL) - { - ephy_bookmarks_add_keyword (entry->priv->bookmarks, action); - } + EphyTopicsEntryPrivate *priv = entry->priv; + char *title; - insert_text (entry, action); - g_free (action); + title = g_strdup (priv->create); + + ephy_bookmarks_add_keyword (priv->bookmarks, title); + update_widget (entry); + + insert_text (entry, title); + g_free (title); } static gboolean @@ -358,22 +457,15 @@ match_selected_cb (GtkEntryCompletion *completion, gpointer user_data) { EphyTopicsEntry *entry = EPHY_TOPICS_ENTRY (gtk_entry_completion_get_entry (completion)); - char *title; + GValue value = { 0, }; - gtk_tree_model_get (model, iter, 0, &title, -1); - insert_text (entry, title); - g_free (title); + gtk_tree_model_get_value (model, iter, COLUMN_TITLE, &value); + insert_text (entry, g_value_get_string (&value)); + g_value_unset (&value); return TRUE; } -static void -tree_changed_cb (EphyBookmarks *bookmarks, - EphyTopicsEntry *entry) -{ - update_widget(entry); -} - static gboolean focus_out_cb (GtkEditable *editable, GdkEventFocus *event, @@ -383,40 +475,59 @@ focus_out_cb (GtkEditable *editable, return FALSE; } -static gboolean -match_func (GtkEntryCompletion *completion, - const gchar *key, - GtkTreeIter *iter, - gpointer user_data) +static void +tree_changed_cb (EphyBookmarks *bookmarks, + EphyTopicsEntry *entry) { - GtkEntry *entry = gtk_entry_completion_get_entry (completion); - GtkTreeModel *model = gtk_entry_completion_get_model (completion); - EphyTopicsEntryPrivate *priv = EPHY_TOPICS_ENTRY(entry)->priv; - - gboolean result; - char *text; + update_widget (entry); +} - if (priv->key == NULL) return TRUE; - gtk_tree_model_get (model, iter, 1, &text, -1); - if (text == NULL) return FALSE; - result = g_str_has_prefix (text, priv->key); - g_free (text); - - return result; +static void +node_added_cb (EphyNode *parent, + EphyNode *child, + GObject *object) +{ + update_widget (EPHY_TOPICS_ENTRY (object)); +} + +static void +node_changed_cb (EphyNode *parent, + EphyNode *child, + guint property_id, + GObject *object) +{ + update_widget (EPHY_TOPICS_ENTRY (object)); +} + +static void +node_removed_cb (EphyNode *parent, + EphyNode *child, + guint index, + GObject *object) +{ + update_widget (EPHY_TOPICS_ENTRY (object)); } static void ephy_topics_entry_set_property (GObject *object, - guint prop_id, - const GValue *value, - GParamSpec *pspec) + guint prop_id, + const GValue *value, + GParamSpec *pspec) { EphyTopicsEntry *entry = EPHY_TOPICS_ENTRY (object); + EphyNode *node; switch (prop_id) { case PROP_BOOKMARKS: entry->priv->bookmarks = g_value_get_object (value); + node = ephy_bookmarks_get_keywords (entry->priv->bookmarks); + ephy_node_signal_connect_object (node, EPHY_NODE_CHILD_ADDED, + (EphyNodeCallback) node_added_cb, object); + ephy_node_signal_connect_object (node, EPHY_NODE_CHILD_CHANGED, + (EphyNodeCallback) node_changed_cb, object); + ephy_node_signal_connect_object (node, EPHY_NODE_CHILD_REMOVED, + (EphyNodeCallback) node_removed_cb, object); g_signal_connect_object (entry->priv->bookmarks, "tree-changed", G_CALLBACK (tree_changed_cb), entry, G_CONNECT_AFTER); @@ -444,12 +555,11 @@ ephy_topics_entry_constructor (GType type, entry = EPHY_TOPICS_ENTRY (object); priv = EPHY_TOPICS_ENTRY_GET_PRIVATE (object); - priv->store = gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_STRING); + priv->store = gtk_list_store_new (3, G_TYPE_POINTER, G_TYPE_STRING, G_TYPE_STRING); priv->completion = gtk_entry_completion_new (); - priv->update_keywords = TRUE; gtk_entry_completion_set_model (priv->completion, GTK_TREE_MODEL (priv->store)); - gtk_entry_completion_set_text_column (priv->completion, 0); + gtk_entry_completion_set_text_column (priv->completion, COLUMN_TITLE); gtk_entry_completion_set_popup_completion (priv->completion, TRUE); gtk_entry_completion_set_popup_single_match (priv->completion, TRUE); gtk_entry_completion_set_match_func (priv->completion, match_func, NULL, NULL); @@ -463,12 +573,12 @@ ephy_topics_entry_constructor (GType type, g_signal_connect (object, "focus-out-event", G_CALLBACK (focus_out_cb), NULL); g_signal_connect (object, "changed", - G_CALLBACK (update_keywords), NULL); + G_CALLBACK (update_database), NULL); g_signal_connect (object, "notify::cursor-position", - G_CALLBACK (update_input), NULL); + G_CALLBACK (update_key), NULL); g_signal_connect (object, "notify::text", - G_CALLBACK (update_input), NULL); + G_CALLBACK (update_key), NULL); update_widget (entry); @@ -487,7 +597,7 @@ ephy_topics_entry_finalize (GObject *object) { EphyTopicsEntry *entry = EPHY_TOPICS_ENTRY (object); - g_free (entry->priv->input); + g_free (entry->priv->create); g_free (entry->priv->key); parent_class->finalize (object); |