diff options
-rw-r--r-- | widgets/text/e-completion-callbacks.c | 96 | ||||
-rw-r--r-- | widgets/text/e-completion-callbacks.h | 68 | ||||
-rw-r--r-- | widgets/text/e-completion-match.c | 1 | ||||
-rw-r--r-- | widgets/text/e-completion-match.h | 1 | ||||
-rw-r--r-- | widgets/text/e-completion-test.c | 220 | ||||
-rw-r--r-- | widgets/text/e-completion-view.c | 94 | ||||
-rw-r--r-- | widgets/text/e-completion-view.h | 4 | ||||
-rw-r--r-- | widgets/text/e-completion.c | 363 | ||||
-rw-r--r-- | widgets/text/e-completion.h | 21 | ||||
-rw-r--r-- | widgets/text/e-entry.c | 78 | ||||
-rw-r--r-- | widgets/text/e-text.c | 6 |
11 files changed, 239 insertions, 713 deletions
diff --git a/widgets/text/e-completion-callbacks.c b/widgets/text/e-completion-callbacks.c new file mode 100644 index 0000000000..935ba7fb43 --- /dev/null +++ b/widgets/text/e-completion-callbacks.c @@ -0,0 +1,96 @@ +/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ +/* + * e-completion-callbacks.c - A callback based ECompletion. + * Copyright 2003 + * + * Authors: + * Chris Toshok <toshok@ximian.com> + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License, version 2, as published by the Free Software Foundation. + * + * This library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA + * 02111-1307, USA. + */ + +#include <config.h> +#include <string.h> +#include <stdio.h> +#include <gtk/gtk.h> +#include "gal/util/e-util.h" +#include "e-completion-callbacks.h" + +static void e_completion_callbacks_class_init (ECompletionCallbacksClass *klass); +static void e_completion_callbacks_init (ECompletionCallbacks *complete); + +static void callbacks_request_completion (ECompletion *comp, const gchar *search_text, gint pos, gint limit); +static void callbacks_end_completion (ECompletion *comp); + +#define PARENT_TYPE E_COMPLETION_TYPE +static ECompletionClass *parent_class; + + + +E_MAKE_TYPE (e_completion_callbacks, + "ECompletionCallbacks", + ECompletionCallbacks, + e_completion_callbacks_class_init, + e_completion_callbacks_init, + PARENT_TYPE) + +static void +e_completion_callbacks_class_init (ECompletionCallbacksClass *klass) +{ + ECompletionClass *comp_class = (ECompletionClass *) klass; + + parent_class = g_type_class_ref (PARENT_TYPE); + + comp_class->request_completion = callbacks_request_completion; + comp_class->end_completion = callbacks_end_completion; +} + +static void +e_completion_callbacks_init (ECompletionCallbacks *complete) +{ +} + +static void +callbacks_request_completion (ECompletion *comp, const gchar *search_text, gint pos, gint limit) +{ + ECompletionCallbacks *cc = E_COMPLETION_CALLBACKS (comp); + + cc->request_completion (cc, search_text, pos, limit, cc->data); +} + +static void +callbacks_end_completion (ECompletion *comp) +{ + ECompletionCallbacks *cc = E_COMPLETION_CALLBACKS (comp); + + cc->end_completion (cc, cc->data); +} + +ECompletion* +e_completion_callbacks_new (ECompletionCallbacksRequestCompletionFn request_completion, + ECompletionCallbacksEndCompletionFn end_completion, + gpointer data) +{ + ECompletionCallbacks *cc; + + g_return_val_if_fail (request_completion != NULL, NULL); + g_return_val_if_fail (end_completion != NULL, NULL); + + cc = gtk_type_new (E_COMPLETION_CALLBACKS_TYPE); + + cc->request_completion = request_completion; + cc->end_completion = end_completion; + cc->data = data; +} diff --git a/widgets/text/e-completion-callbacks.h b/widgets/text/e-completion-callbacks.h new file mode 100644 index 0000000000..57df9cd3e5 --- /dev/null +++ b/widgets/text/e-completion-callbacks.h @@ -0,0 +1,68 @@ +/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ +/* + * e-completion-callback.h - A callback based completion object. + * Copyright 2003, Ximian, Inc. + * + * Authors: + * Chris Toshok <toshok@ximian.com> + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License, version 2, as published by the Free Software Foundation. + * + * This library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA + * 02111-1307, USA. + */ + +#ifndef E_COMPLETION_CALLBACKS_H +#define E_COMPLETION_CALLBACKS_H + +#include <gtk/gtkobject.h> +#include "e-completion.h" + +G_BEGIN_DECLS + +#define E_COMPLETION_CALLBACKS_TYPE (e_completion_callbacks_get_type ()) +#define E_COMPLETION_CALLBACKS(o) (G_TYPE_CHECK_INSTANCE_CAST ((o), E_COMPLETION_CALLBACKS_TYPE, ECompletionCallbacks)) +#define E_COMPLETION_CALLBACKS_CLASS(k) (G_TYPE_CHECK_CLASS_CAST ((k), E_COMPLETION_CALLBACKS_TYPE, ECompletionCallbacksClass)) +#define E_IS_COMPLETION_CALLBACKS(o) (G_TYPE_CHECK_INSTANCE_TYPE ((o), E_COMPLETION_CALLBACKS_TYPE)) +#define E_IS_COMPLETION_CALLBACKS_CLASS(k) (G_TYPE_CHECK_CLASS_TYPE ((k), E_COMPLETION_CALLBACKS_TYPE)) + +typedef struct _ECompletionCallbacks ECompletionCallbacks; +typedef struct _ECompletionCallbacksClass ECompletionCallbacksClass; +struct _ECompletionCallbacksPrivate; + +typedef void (*ECompletionCallbacksRequestCompletionFn) (ECompletionCallbacks *comp, const gchar *search_text, gint pos, gint limit, gpointer data); +typedef void (*ECompletionCallbacksEndCompletionFn) (ECompletionCallbacks *comp, gpointer data); + +struct _ECompletionCallbacks { + ECompletion parent; + + ECompletionCallbacksRequestCompletionFn request_completion; + ECompletionCallbacksEndCompletionFn end_completion; + + gpointer data; +}; + +struct _ECompletionCallbacksClass { + ECompletionClass parent_class; +}; + +GtkType e_completion_callbacks_get_type (void); + +ECompletion* e_completion_callbacks_new (ECompletionCallbacksRequestCompletionFn request_completion, + ECompletionCallbacksEndCompletionFn end_completion, + gpointer data); + +G_END_DECLS + + +#endif /* E_COMPLETION_CALLBACKS_H */ + diff --git a/widgets/text/e-completion-match.c b/widgets/text/e-completion-match.c index 0da197d1fe..d13ba15973 100644 --- a/widgets/text/e-completion-match.c +++ b/widgets/text/e-completion-match.c @@ -50,7 +50,6 @@ e_completion_match_construct (ECompletionMatch *match) match->sort_minor = 0; match->user_data = NULL; match->ref = 1; - match->hit_count = 0; match->destroy = NULL; } diff --git a/widgets/text/e-completion-match.h b/widgets/text/e-completion-match.h index f2773ee094..162373add4 100644 --- a/widgets/text/e-completion-match.h +++ b/widgets/text/e-completion-match.h @@ -40,7 +40,6 @@ struct _ECompletionMatch { gpointer user_data; gint ref; - gint hit_count; void (*destroy) (ECompletionMatch *); }; diff --git a/widgets/text/e-completion-test.c b/widgets/text/e-completion-test.c deleted file mode 100644 index dfefe23e1b..0000000000 --- a/widgets/text/e-completion-test.c +++ /dev/null @@ -1,220 +0,0 @@ -/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ -/* - * e-completion-test.c - * Copyright 2000, 2001, Ximian, Inc. - * - * Authors: - * Jon Trowbridge <trow@ximian.com> - * - * This library is free software; you can redistribute it and/or - * modify it under the terms of the GNU Library General Public - * License, version 2, as published by the Free Software Foundation. - * - * This library 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 - * Library General Public License for more details. - * - * You should have received a copy of the GNU Library General Public - * License along with this library; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA - * 02111-1307, USA. - */ - -#include <gnome.h> -#include "e-completion.h" -#include "e-entry.h" - -#define TIMEOUT 10 - -/* Dictionary Lookup test */ - -static gint word_count = 0; -static gchar **word_array = NULL; - -static void -read_dict (void) -{ - FILE *in = fopen ("/usr/share/dict/words", "r"); - gchar buffer[128]; - GList *word_list = NULL, *iter; - gint i; - - while (fgets (buffer, 128, in)) { - gint len = strlen (buffer); - if (len > 0 && buffer[len-1] == '\n') - buffer[len-1] = '\0'; - word_list = g_list_prepend (word_list, g_strdup (buffer)); - ++word_count; - } - fclose (in); - - word_array = g_new (gchar *, word_count); - i = word_count-1; - for (iter = word_list; iter != NULL; iter = g_list_next (iter)) { - word_array[i] = (gchar *)iter->data; - --i; - } -} - -static gint -find_word (const gchar *str) -{ - gint a, b; - - if (word_array == NULL) - read_dict (); - - a = 0; - b = word_count-1; - - while (b-a > 1) { - gint m = (a+b)/2; - gint cmp = g_strcasecmp (str, word_array[m]); - - if (cmp < 0) - b = m; - else if (cmp > 0) - a = m; - else - return m; - } - - return b; -} - -struct { - ECompletion *complete; - const gchar *txt; - gint start; - gint current; - gint len; - gint limit; - gint count; -} dict_info; -static guint dict_tag = 0; - -static gboolean -dict_check (gpointer ptr) -{ - gint limit = dict_info.limit; - gint i; - - /* If this is the first iteration, do the binary search in our word list to figure out - where to start. We do less work on the first iteration, to give more of a sense of - immediate feedback. */ - if (dict_info.start < 0) { - dict_info.start = dict_info.current = find_word (dict_info.txt); - } - - i = dict_info.current; - while (limit > 0 - && i < word_count - && dict_info.count < 50 - && g_strncasecmp (dict_info.txt, word_array[i], dict_info.len) == 0) { - - ECompletionMatch *match = g_new (ECompletionMatch, 1); - e_completion_match_construct (match); - e_completion_match_set_text (match, word_array[i], NULL); - match->score = dict_info.len / (double)strlen (word_array[i]); - e_completion_found_match (dict_info.complete, match); - - ++i; - --limit; - ++dict_info.count; - } - dict_info.current = i; - dict_info.limit = MIN (dict_info.limit*2, 400); - - if (limit != 0) { - dict_tag = 0; - e_completion_end_search (dict_info.complete); - return FALSE; - } - - - - return TRUE; -} - -static void -request_dict_search (ECompletion *complete, const gchar *txt, gint pos, gint limit, gpointer user_data) -{ - gint len = strlen (txt); - - if (dict_tag != 0) { - gtk_timeout_remove (dict_tag); - dict_tag = 0; - } - - if (len > 0) { - dict_info.complete = complete; - dict_info.txt = txt; - dict_info.start = -1; - dict_info.current = -1; - dict_info.len = len; - dict_info.limit = 100; - dict_info.count = 0; - dict_tag = gtk_timeout_add (TIMEOUT, dict_check, NULL); - } else { - e_completion_end_search (complete); - } -} - -static void -end_dict_search (ECompletion *complete, gpointer user_data) -{ - if (dict_tag != 0) { - gtk_timeout_remove (dict_tag); - dict_tag = 0; - } -} - -static void -popup_cb (EEntry *popup, GdkEventButton *ev, gint pos, gpointer user_data) -{ - g_print ("popup at pos %d\n", pos); -} - -int -main (int argc, gchar **argv) -{ - ECompletion* complete; - GtkWidget *entry; - GtkWidget *win; - - gnome_init ("ETextModelTest", "0.0", argc, argv); - - read_dict (); - - complete = e_completion_new (); - g_signal_connect (complete, - "request_completion", - G_CALLBACK (request_dict_search), - NULL); - g_signal_connect (complete, - "end_completion", - G_CALLBACK (end_dict_search), - NULL); - g_signal_connect (complete, - "cancel_completion", - G_CALLBACK (end_dict_search), - NULL); - - win = gtk_window_new (GTK_WINDOW_TOPLEVEL); - entry = e_entry_new (); - e_entry_enable_completion_full (E_ENTRY (entry), complete, 0, NULL); - e_entry_set_editable (E_ENTRY (entry), TRUE); - - g_signal_connect (entry, - "popup", - G_CALLBACK (popup_cb), - NULL); - - gtk_container_add (GTK_CONTAINER (win), entry); - gtk_widget_show_all (win); - - gtk_main (); - - return 0; -} diff --git a/widgets/text/e-completion-view.c b/widgets/text/e-completion-view.c index 774e73cd17..f9728de60c 100644 --- a/widgets/text/e-completion-view.c +++ b/widgets/text/e-completion-view.c @@ -161,6 +161,8 @@ e_completion_view_size_request (GtkWidget *widget, GtkRequisition *requisition) requisition->width += child_requisition.width; requisition->height += child_requisition.height; } + + requisition->height = MAX (100, requisition->height); } static void @@ -319,24 +321,12 @@ e_completion_view_disconnect (ECompletionView *cv) g_signal_handler_disconnect (cv->completion, cv->begin_signal_id); if (cv->comp_signal_id) g_signal_handler_disconnect (cv->completion, cv->comp_signal_id); - if (cv->restart_signal_id) - g_signal_handler_disconnect (cv->completion, cv->restart_signal_id); - if (cv->cancel_signal_id) - g_signal_handler_disconnect (cv->completion, cv->cancel_signal_id); if (cv->end_signal_id) g_signal_handler_disconnect (cv->completion, cv->end_signal_id); - if (cv->clear_signal_id) - g_signal_handler_disconnect (cv->completion, cv->clear_signal_id); - if (cv->lost_signal_id) - g_signal_handler_disconnect (cv->completion, cv->lost_signal_id); cv->begin_signal_id = 0; cv->comp_signal_id = 0; - cv->restart_signal_id = 0; - cv->cancel_signal_id = 0; cv->end_signal_id = 0; - cv->clear_signal_id = 0; - cv->lost_signal_id = 0; } static ETable * @@ -594,28 +584,6 @@ begin_completion_cb (ECompletion *completion, const gchar *txt, gint pos, gint l } static void -restart_completion_cb (ECompletion *completion, gpointer user_data) -{ - /* For now, handle restarts like the beginning of a new completion. */ - begin_completion_cb (completion, NULL, 0, 0, user_data); -} - -static void -cancel_completion_cb (ECompletion *completion, gpointer user_data) -{ - ECompletionView *cv = E_COMPLETION_VIEW (user_data); - - /* On a cancel, clear our choices and issue an "unbrowse" signal. */ - e_table_model_pre_change (cv->model); - e_completion_view_clear_choices (cv); - cv->have_all_choices = TRUE; - e_completion_view_set_cursor_row (cv, -1); - e_table_model_changed (cv->model); - - g_signal_emit (cv, e_completion_view_signals[E_COMPLETION_VIEW_UNBROWSE] ,0); -} - -static void completion_cb (ECompletion *completion, ECompletionMatch *match, gpointer user_data) { ECompletionView *cv = E_COMPLETION_VIEW (user_data); @@ -648,42 +616,10 @@ end_completion_cb (ECompletion *completion, gpointer user_data) g_signal_emit (cv, e_completion_view_signals[E_COMPLETION_VIEW_FULL], 0); } -static void -clear_completion_cb (ECompletion *completion, gpointer user_data) -{ - ECompletionView *cv = E_COMPLETION_VIEW (user_data); - - e_table_model_pre_change (cv->model); - e_completion_view_clear_choices (cv); - cv->have_all_choices = FALSE; - - e_table_model_changed (cv->model); -} - -static void -lost_completion_cb (ECompletion *completion, ECompletionMatch *match, gpointer user_data) -{ - ECompletionView *cv = E_COMPLETION_VIEW (user_data); - int i; - GPtrArray *c = cv->choices; - - for (i = 0; i < c->len; i++) - if (g_ptr_array_index (c, i) == match) - break; - - g_return_if_fail (i == c->len); - - /* FIXME: do remove_index_fast(), then row_changed and - * row_deleted (if there are more than 1 row still) */ - e_table_model_pre_change (cv->model); - g_ptr_array_remove_index (c, i); - e_table_model_row_deleted (cv->model, i); - - e_completion_match_unref (match); -} - /*** Table Callbacks ***/ +/* XXX toshok - we need to add sorting to this etable, through the use + of undisplayed fields of all the sort keys we want to use */ static char *simple_spec = "<ETableSpecification no-headers=\"true\" draw-grid=\"false\" cursor-mode=\"line\" alternating-row-colors=\"false\" gettext-domain=\"" E_I18N_DOMAIN "\">" " <ETableColumn model_col=\"0\" _title=\"Node\" expansion=\"1.0\" " @@ -756,33 +692,17 @@ e_completion_view_construct (ECompletionView *cv, ECompletion *completion) g_object_ref (completion); cv->begin_signal_id = g_signal_connect (completion, - "begin_completion", + "completion_started", G_CALLBACK (begin_completion_cb), cv); cv->comp_signal_id = g_signal_connect (completion, - "completion", + "completion_found", G_CALLBACK (completion_cb), cv); - cv->restart_signal_id = g_signal_connect (completion, - "restart_completion", - G_CALLBACK (restart_completion_cb), - cv); - cv->cancel_signal_id = g_signal_connect (completion, - "cancel_completion", - G_CALLBACK (cancel_completion_cb), - cv); cv->end_signal_id = g_signal_connect (completion, - "end_completion", + "completion_finished", G_CALLBACK (end_completion_cb), cv); - cv->clear_signal_id = g_signal_connect (completion, - "clear_completion", - G_CALLBACK (clear_completion_cb), - cv); - cv->lost_signal_id = g_signal_connect (completion, - "lost_completion", - G_CALLBACK (lost_completion_cb), - cv); cv->model = e_table_simple_new (table_col_count, table_row_count, diff --git a/widgets/text/e-completion-view.h b/widgets/text/e-completion-view.h index 9aa3860ebd..25b2eff645 100644 --- a/widgets/text/e-completion-view.h +++ b/widgets/text/e-completion-view.h @@ -51,11 +51,7 @@ struct _ECompletionView { ECompletion *completion; guint begin_signal_id; guint comp_signal_id; - guint restart_signal_id; - guint cancel_signal_id; guint end_signal_id; - guint clear_signal_id; - guint lost_signal_id; GtkWidget *key_widget; guint key_signal_id; diff --git a/widgets/text/e-completion.c b/widgets/text/e-completion.c index cddd6ed4f4..b243d4db35 100644 --- a/widgets/text/e-completion.c +++ b/widgets/text/e-completion.c @@ -31,47 +31,32 @@ #include "gal/util/e-marshal.h" enum { - E_COMPLETION_REQUEST_COMPLETION, - E_COMPLETION_BEGIN_COMPLETION, - E_COMPLETION_COMPLETION, - E_COMPLETION_RESTART_COMPLETION, - E_COMPLETION_CANCEL_COMPLETION, - E_COMPLETION_END_COMPLETION, - E_COMPLETION_CLEAR_COMPLETION, - E_COMPLETION_LOST_COMPLETION, - E_COMPLETION_LAST_SIGNAL + COMPLETION_STARTED, + COMPLETION_FOUND, + COMPLETION_CANCELED, + COMPLETION_FINISHED, + LAST_SIGNAL }; -static guint e_completion_signals[E_COMPLETION_LAST_SIGNAL] = { 0 }; +static guint e_completion_signals[LAST_SIGNAL] = { 0 }; struct _ECompletionPrivate { gboolean searching; gboolean done_search; - gboolean refining; gchar *search_text; GPtrArray *matches; - gint match_count; gint pos; gint limit; double min_score, max_score; - gint refinement_count; - GList *search_stack; }; -typedef struct { - gchar *text; - gint pos; -} ECompletionSearch; - static void e_completion_class_init (ECompletionClass *klass); static void e_completion_init (ECompletion *complete); static void e_completion_dispose (GObject *object); static void e_completion_add_match (ECompletion *complete, ECompletionMatch *); -static void e_completion_clear_search_stack (ECompletion *complete); static void e_completion_clear_matches (ECompletion *complete); static gboolean e_completion_sort (ECompletion *complete); -static void e_completion_restart (ECompletion *complete); #define PARENT_TYPE GTK_TYPE_OBJECT static GtkObjectClass *parent_class; @@ -92,80 +77,34 @@ e_completion_class_init (ECompletionClass *klass) parent_class = g_type_class_ref (PARENT_TYPE); - e_completion_signals[E_COMPLETION_REQUEST_COMPLETION] = - g_signal_new ("request_completion", - G_OBJECT_CLASS_TYPE (object_class), - G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ECompletionClass, request_completion), - NULL, NULL, - e_marshal_NONE__POINTER_INT_INT, - G_TYPE_NONE, 3, - G_TYPE_POINTER, G_TYPE_INT, G_TYPE_INT); - - e_completion_signals[E_COMPLETION_BEGIN_COMPLETION] = - g_signal_new ("begin_completion", + e_completion_signals[COMPLETION_STARTED] = + g_signal_new ("completion_started", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ECompletionClass, begin_completion), + G_STRUCT_OFFSET (ECompletionClass, completion_started), NULL, NULL, e_marshal_NONE__POINTER_INT_INT, G_TYPE_NONE, 3, G_TYPE_POINTER, G_TYPE_INT, G_TYPE_INT); - e_completion_signals[E_COMPLETION_COMPLETION] = - g_signal_new ("completion", + e_completion_signals[COMPLETION_FOUND] = + g_signal_new ("completion_found", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ECompletionClass, completion), + G_STRUCT_OFFSET (ECompletionClass, completion_found), NULL, NULL, e_marshal_NONE__POINTER, G_TYPE_NONE, 1, G_TYPE_POINTER); - e_completion_signals[E_COMPLETION_RESTART_COMPLETION] = - g_signal_new ("restart_completion", - G_OBJECT_CLASS_TYPE (object_class), - G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ECompletionClass, restart_completion), - NULL, NULL, - e_marshal_NONE__NONE, - G_TYPE_NONE, 0); - - e_completion_signals[E_COMPLETION_CANCEL_COMPLETION] = - g_signal_new ("cancel_completion", + e_completion_signals[COMPLETION_FINISHED] = + g_signal_new ("completion_finished", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ECompletionClass, cancel_completion), + G_STRUCT_OFFSET (ECompletionClass, completion_finished), NULL, NULL, e_marshal_NONE__NONE, G_TYPE_NONE, 0); - - e_completion_signals[E_COMPLETION_END_COMPLETION] = - g_signal_new ("end_completion", - G_OBJECT_CLASS_TYPE (object_class), - G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ECompletionClass, end_completion), - NULL, NULL, - e_marshal_NONE__NONE, - G_TYPE_NONE, 0); - - e_completion_signals[E_COMPLETION_CLEAR_COMPLETION] = - g_signal_new ("clear_completion", - G_OBJECT_CLASS_TYPE (object_class), - G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ECompletionClass, clear_completion), - NULL, NULL, - e_marshal_NONE__NONE, - G_TYPE_NONE, 0); - - e_completion_signals[E_COMPLETION_LOST_COMPLETION] = - g_signal_new ("lost_completion", - G_OBJECT_CLASS_TYPE (object_class), - G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ECompletionClass, lost_completion), - NULL, NULL, - e_marshal_NONE__POINTER, - G_TYPE_NONE, 1, G_TYPE_POINTER); object_class->dispose = e_completion_dispose; } @@ -187,7 +126,6 @@ e_completion_dispose (GObject *object) complete->priv->search_text = NULL; e_completion_clear_matches (complete); - e_completion_clear_search_stack (complete); g_ptr_array_free (complete->priv->matches, TRUE); complete->priv->matches = NULL; @@ -241,149 +179,9 @@ e_completion_clear_matches (ECompletion *complete) } void -e_completion_clear (ECompletion *complete) -{ - g_return_if_fail (E_IS_COMPLETION (complete)); - - /* FIXME: do we really want _clear and _clear_matches() ? */ - - /* I think yes, because it is convenient to be able to clear our match cache - without emitting a "clear_completion" signal. -JT */ - - e_completion_clear_matches (complete); - e_completion_clear_search_stack (complete); - complete->priv->refinement_count = 0; - complete->priv->match_count = 0; - g_signal_emit (complete, e_completion_signals[E_COMPLETION_CLEAR_COMPLETION], 0); -} - -static void -e_completion_push_search (ECompletion *complete, const gchar *text, gint pos) -{ - ECompletionSearch *search; - - g_return_if_fail (E_IS_COMPLETION (complete)); - - search = g_new (ECompletionSearch, 1); - search->text = complete->priv->search_text; - search->pos = complete->priv->pos; - complete->priv->search_stack = g_list_prepend (complete->priv->search_stack, search); - - complete->priv->search_text = g_strdup (text); - complete->priv->pos = pos; -} - -static void -e_completion_pop_search (ECompletion *complete) -{ - ECompletionSearch *search; - GList *old_link = complete->priv->search_stack; - - g_return_if_fail (E_IS_COMPLETION (complete)); - g_return_if_fail (complete->priv->search_stack != NULL); - - g_free (complete->priv->search_text); - - search = complete->priv->search_stack->data; - complete->priv->search_text = search->text; - complete->priv->pos = search->pos; - - g_free (search); - complete->priv->search_stack = g_list_remove_link (complete->priv->search_stack, - complete->priv->search_stack); - g_list_free_1 (old_link); -} - -static void -e_completion_clear_search_stack (ECompletion *complete) -{ - GList *iter; - - g_return_if_fail (E_IS_COMPLETION (complete)); - - for (iter = complete->priv->search_stack; iter != NULL; iter = g_list_next (iter)) { - ECompletionSearch *search = iter->data; - g_free (search->text); - g_free (search); - } - g_list_free (complete->priv->search_stack); - complete->priv->search_stack = NULL; -} - -static void -e_completion_refine_search (ECompletion *comp, const gchar *text, gint pos, ECompletionRefineFn refine_fn) -{ - GPtrArray *m; - gint i; - - comp->priv->refining = TRUE; - - e_completion_push_search (comp, text, pos); - - g_signal_emit (comp, e_completion_signals[E_COMPLETION_BEGIN_COMPLETION], 0, text, pos, comp->priv->limit); - - comp->priv->match_count = 0; - - comp->priv->searching = TRUE; - - m = comp->priv->matches; - for (i = 0; i < m->len; ++i) { - ECompletionMatch *match = g_ptr_array_index (m, i); - if (comp->priv->refinement_count == match->hit_count - && refine_fn (comp, match, text, pos)) { - ++match->hit_count; - g_signal_emit (comp, e_completion_signals[E_COMPLETION_COMPLETION], 0, match); - ++comp->priv->match_count; - } - } - - ++comp->priv->refinement_count; - - g_signal_emit (comp, e_completion_signals[E_COMPLETION_END_COMPLETION], 0); - - comp->priv->searching = FALSE; - comp->priv->refining = FALSE; -} - -static void -e_completion_unrefine_search (ECompletion *comp) -{ - GPtrArray *m; - gint i; - - comp->priv->refining = TRUE; - - e_completion_pop_search (comp); - - g_signal_emit (comp, e_completion_signals[E_COMPLETION_BEGIN_COMPLETION], 0, - comp->priv->search_text, comp->priv->pos, comp->priv->limit); - - comp->priv->match_count = 0; - --comp->priv->refinement_count; - - comp->priv->searching = TRUE; - - m = comp->priv->matches; - for (i = 0; i < m->len; ++i) { - ECompletionMatch *match = g_ptr_array_index (m, i); - if (comp->priv->refinement_count <= match->hit_count) { - match->hit_count = comp->priv->refinement_count; - g_signal_emit (comp, e_completion_signals[E_COMPLETION_COMPLETION], 0, match); - ++comp->priv->match_count; - } - } - - g_signal_emit (comp, e_completion_signals[E_COMPLETION_END_COMPLETION], 0); - - comp->priv->searching = FALSE; - comp->priv->refining = FALSE; -} - -void e_completion_begin_search (ECompletion *complete, const gchar *text, gint pos, gint limit) { ECompletionClass *klass; - ECompletionRefineFn refine_fn; g_return_if_fail (complete != NULL); g_return_if_fail (E_IS_COMPLETION (complete)); @@ -391,35 +189,6 @@ e_completion_begin_search (ECompletion *complete, const gchar *text, gint pos, g klass = E_COMPLETION_CLASS (GTK_OBJECT_GET_CLASS (complete)); - if (!complete->priv->searching && complete->priv->done_search) { - - /* If the search we are requesting is the same as what we had before our last refinement, - treat the request as an unrefine. */ - if (complete->priv->search_stack != NULL) { - ECompletionSearch *search = complete->priv->search_stack->data; - if ((klass->ignore_pos_on_auto_unrefine || search->pos == pos) - && !strcmp (search->text, text)) { - e_completion_unrefine_search (complete); - return; - } - } - - if (klass->auto_refine - && (refine_fn = klass->auto_refine (complete, - complete->priv->search_text, complete->priv->pos, - text, pos))) { - e_completion_refine_search (complete, text, pos, refine_fn); - return; - } - - } - - /* Stop any prior search. */ - if (complete->priv->searching) - e_completion_cancel_search (complete); - - e_completion_clear_search_stack (complete); - g_free (complete->priv->search_text); complete->priv->search_text = g_strdup (text); @@ -430,25 +199,10 @@ e_completion_begin_search (ECompletion *complete, const gchar *text, gint pos, g e_completion_clear_matches (complete); complete->priv->limit = limit > 0 ? limit : G_MAXINT; - complete->priv->refinement_count = 0; - - g_signal_emit (complete, e_completion_signals[E_COMPLETION_BEGIN_COMPLETION], 0, text, pos, limit); - g_signal_emit (complete, e_completion_signals[E_COMPLETION_REQUEST_COMPLETION], 0, text, pos, limit); -} - -void -e_completion_cancel_search (ECompletion *complete) -{ - g_return_if_fail (complete != NULL); - g_return_if_fail (E_IS_COMPLETION (complete)); - - /* If there is no search to cancel, just silently return. */ - if (!complete->priv->searching) - return; - - g_signal_emit (complete, e_completion_signals[E_COMPLETION_CANCEL_COMPLETION], 0); - complete->priv->searching = FALSE; + g_signal_emit (complete, e_completion_signals[COMPLETION_STARTED], 0, text, pos, limit); + if (klass->request_completion) + klass->request_completion (complete, text, pos, limit); } gboolean @@ -460,15 +214,6 @@ e_completion_searching (ECompletion *complete) return complete->priv->searching; } -gboolean -e_completion_refining (ECompletion *complete) -{ - g_return_val_if_fail (complete != NULL, FALSE); - g_return_val_if_fail (E_IS_COMPLETION (complete), FALSE); - - return complete->priv->refining; -} - const gchar * e_completion_search_text (ECompletion *complete) { @@ -493,7 +238,7 @@ e_completion_match_count (ECompletion *complete) g_return_val_if_fail (complete != NULL, 0); g_return_val_if_fail (E_IS_COMPLETION (complete), 0); - return complete->priv->refinement_count > 0 ? complete->priv->match_count : complete->priv->matches->len; + return complete->priv->matches->len; } void @@ -511,9 +256,7 @@ e_completion_foreach_match (ECompletion *complete, ECompletionMatchFn fn, gpoint m = complete->priv->matches; for (i = 0; i < m->len; i++) { ECompletionMatch *match = g_ptr_array_index (m, i); - if (match->hit_count == complete->priv->refinement_count) { - fn (match, closure); - } + fn (match, closure); } } @@ -554,25 +297,6 @@ e_completion_sort (ECompletion *complete) return diff; } -/* Emit a restart signal and re-declare our matches, up to the limit. */ -static void -e_completion_restart (ECompletion *complete) -{ - GPtrArray *m; - gint i, count; - - g_signal_emit (complete, e_completion_signals[E_COMPLETION_RESTART_COMPLETION], 0); - - m = complete->priv->matches; - for (i = count = 0; - i < m->len && count < complete->priv->limit; - i++, count++) { - g_signal_emit (complete, - e_completion_signals[E_COMPLETION_COMPLETION], 0, - g_ptr_array_index (m, i)); - } -} - void e_completion_found_match (ECompletion *complete, ECompletionMatch *match) { @@ -594,47 +318,22 @@ e_completion_found_match (ECompletion *complete, ECompletionMatch *match) e_completion_add_match (complete, match); - g_signal_emit (complete, e_completion_signals[E_COMPLETION_COMPLETION], 0, match); -} - -/* to optimize this, make the match a hash table */ -void -e_completion_lost_match (ECompletion *complete, ECompletionMatch *match) -{ - gboolean removed; - - g_return_if_fail (E_IS_COMPLETION (complete)); - g_return_if_fail (match != NULL); - - /* FIXME: remove fast */ - removed = g_ptr_array_remove (complete->priv->matches, - match); - - /* maybe just return here? */ - g_return_if_fail (removed); - - g_signal_emit (complete, e_completion_signals[E_COMPLETION_LOST_COMPLETION], 0, match); - - e_completion_match_unref (match); + g_signal_emit (complete, e_completion_signals[COMPLETION_FOUND], 0, match); } void -e_completion_end_search (ECompletion *complete) +e_completion_end_search (ECompletion *comp) { - g_return_if_fail (complete != NULL); - g_return_if_fail (E_IS_COMPLETION (complete)); - g_return_if_fail (complete->priv->searching); - - /* our table model should be sorted by a non-visible column of - * doubles (the score) rather than whatever we are doing - */ - /* If sorting by score accomplishes anything, issue a restart right before we end. */ - if (e_completion_sort (complete)) - e_completion_restart (complete); + g_return_if_fail (comp != NULL); + g_return_if_fail (E_IS_COMPLETION (comp)); + g_return_if_fail (comp->priv->searching); - g_signal_emit (complete, e_completion_signals[E_COMPLETION_END_COMPLETION], 0); + if (E_COMPLETION_CLASS (GTK_OBJECT_GET_CLASS (comp))->end_completion) { + E_COMPLETION_CLASS (GTK_OBJECT_GET_CLASS (comp))->end_completion (comp); + } + g_signal_emit (comp, e_completion_signals[COMPLETION_FINISHED], 0); - complete->priv->searching = FALSE; - complete->priv->done_search = TRUE; + comp->priv->searching = FALSE; + comp->priv->done_search = TRUE; } diff --git a/widgets/text/e-completion.h b/widgets/text/e-completion.h index f8d38e45a7..74976e579c 100644 --- a/widgets/text/e-completion.h +++ b/widgets/text/e-completion.h @@ -52,29 +52,20 @@ struct _ECompletionClass { GtkObjectClass parent_class; /* virtual functions */ - ECompletionRefineFn (*auto_refine) (ECompletion *comp, - const gchar *old_text, gint old_pos, - const gchar *new_text, gint new_pos); - gboolean ignore_pos_on_auto_unrefine; - - /* Signals */ void (*request_completion) (ECompletion *comp, const gchar *search_text, gint pos, gint limit); + void (*end_completion) (ECompletion *comp); - void (*begin_completion) (ECompletion *comp, const gchar *search_text, gint pos, gint limit); - void (*restart_completion) (ECompletion *comp); + /* Signals */ + void (*completion_started) (ECompletion *comp, const gchar *search_text, gint pos, gint limit); - void (*completion) (ECompletion *comp, ECompletionMatch *match); - void (*lost_completion) (ECompletion *comp, ECompletionMatch *match); + void (*completion_found) (ECompletion *comp, ECompletionMatch *match); - void (*cancel_completion) (ECompletion *comp); - void (*end_completion) (ECompletion *comp); - void (*clear_completion) (ECompletion *comp); + void (*completion_finished) (ECompletion *comp); }; GtkType e_completion_get_type (void); void e_completion_begin_search (ECompletion *comp, const gchar *text, gint pos, gint limit); -void e_completion_cancel_search (ECompletion *comp); gboolean e_completion_searching (ECompletion *comp); gboolean e_completion_refining (ECompletion *comp); @@ -91,8 +82,6 @@ ECompletion *e_completion_new (void); or very bad things might happen. */ void e_completion_found_match (ECompletion *comp, ECompletionMatch *); -void e_completion_lost_match (ECompletion *comp, ECompletionMatch *); -void e_completion_clear (ECompletion *comp); void e_completion_end_search (ECompletion *comp); G_END_DECLS diff --git a/widgets/text/e-entry.c b/widgets/text/e-entry.c index a297f1d26a..130b66f02d 100644 --- a/widgets/text/e-entry.c +++ b/widgets/text/e-entry.c @@ -42,15 +42,8 @@ #include "e-text.h" #include "e-entry.h" -#define MOVE_RIGHT_AND_UP 0 - -#define EVIL_POINTER_WARPING_HACK - -#ifdef EVIL_POINTER_WARPING_HACK -#include <gdk/gdkx.h> -#endif - #define MIN_ENTRY_WIDTH 150 +#define INNER_BORDER 2 #define d(x) @@ -166,6 +159,30 @@ canvas_size_allocate (GtkWidget *widget, GtkAllocation *alloc, } static void +get_borders (EEntry *entry, + gint *xborder, + gint *yborder) +{ + GtkWidget *widget = GTK_WIDGET (entry); + gint focus_width; + gboolean interior_focus; + + gtk_widget_style_get (widget, + "interior-focus", &interior_focus, + "focus-line-width", &focus_width, + NULL); + + *xborder = widget->style->xthickness; + *yborder = widget->style->ythickness; + + if (!interior_focus) + { + *xborder += focus_width; + *yborder += focus_width; + } +} + +static void canvas_size_request (GtkWidget *widget, GtkRequisition *requisition, EEntry *entry) { @@ -177,10 +194,8 @@ canvas_size_request (GtkWidget *widget, GtkRequisition *requisition, g_return_if_fail (GNOME_IS_CANVAS (widget)); g_return_if_fail (requisition != NULL); - if (entry->priv->draw_borders) { - xthick = 2 * widget->style->xthickness; - ythick = 2 * widget->style->ythickness; + get_borders (entry, &xthick, &ythick); } else { xthick = ythick = 0; } @@ -190,7 +205,7 @@ canvas_size_request (GtkWidget *widget, GtkRequisition *requisition, g_object_get (entry->item, "text_width", &width, NULL); - requisition->width = 2 + xthick + width; + requisition->width = 2 + 2 * xthick + width; } else { requisition->width = 2 + MIN_ENTRY_WIDTH + xthick; } @@ -207,7 +222,7 @@ canvas_size_request (GtkWidget *widget, GtkRequisition *requisition, requisition->height = (2 + PANGO_PIXELS (pango_font_metrics_get_ascent (metrics) + pango_font_metrics_get_descent (metrics)) + - ythick); + 2 * ythick); pango_font_metrics_unref (metrics); } @@ -287,8 +302,6 @@ e_entry_init (GtkObject *object) entry->priv->emulate_label_resize = FALSE; - entry->priv->emulate_label_resize = FALSE; - entry->canvas = GNOME_CANVAS (e_canvas_new ()); g_signal_connect (entry->canvas, @@ -500,43 +513,12 @@ e_entry_show_popup (EEntry *entry, gboolean visible) x = xo + dim->x; y = yo + dim->height + dim->y; -#if MOVE_RIGHT_AND_UP - /* Put our popup slightly to the right and up, to try to give a visual cue that this popup - is tied to this entry. Otherwise one-row popups can sort of "blend" with an entry - directly below. */ - fudge = MAX (dim->height/10, 3); /* just in case we are using a really big font, etc. */ - x += 2*fudge; - y -= fudge; -#else fudge = 1; y -= fudge; -#endif + gtk_widget_set_uposition (pop, x, y); e_completion_view_set_width (E_COMPLETION_VIEW (entry->priv->completion_view), dim->width); -#ifdef EVIL_POINTER_WARPING_HACK - /* - I should have learned by now to listen to Havoc... - http://developer.gnome.org/doc/GGAD/faqs.html - */ - - if (! entry->priv->popup_is_visible) { - GdkWindow *gwin = GTK_WIDGET (entry)->window; - gint xx, yy; - gdk_window_get_pointer (gwin, &xx, &yy, NULL); - xx += xo; - yy += yo; - - /* If we are inside the "zone of death" where the popup will appear, warp the pointer to safety. - This is a horrible thing to do. */ - if (y <= yy && yy < yy + dim->height && x <= xx && xx < xx + dim->width) { - XWarpPointer (GDK_WINDOW_XDISPLAY (gwin), None, GDK_WINDOW_XWINDOW (gwin), - 0, 0, 0, 0, - xx - xo, (y-1) - yo); - } - } -#endif - gtk_widget_show (pop); @@ -584,8 +566,6 @@ e_entry_start_completion (EEntry *entry) if (entry->priv->completion == NULL) return; - e_entry_cancel_delayed_completion (entry); - if (e_entry_is_empty (entry)) return; diff --git a/widgets/text/e-text.c b/widgets/text/e-text.c index a28b6f2649..26b1534817 100644 --- a/widgets/text/e-text.c +++ b/widgets/text/e-text.c @@ -62,7 +62,7 @@ #define PARENT_TYPE (gnome_canvas_item_get_type()) -#define BORDER_INDENT 4 +#define BORDER_INDENT 3 #define d(x) enum { @@ -1223,8 +1223,8 @@ show_pango_rectangle (EText *text, PangoRectangle rect) } if (clip_height >= 0) { - if (2 + y2 - clip_height > new_yofs_edit) - new_yofs_edit = 2 + y2 - clip_height; + if (y2 - clip_height > new_yofs_edit) + new_yofs_edit = y2 - clip_height; } else { new_yofs_edit = 0; } |