aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--widgets/text/e-completion-callbacks.c96
-rw-r--r--widgets/text/e-completion-callbacks.h68
-rw-r--r--widgets/text/e-completion-match.c1
-rw-r--r--widgets/text/e-completion-match.h1
-rw-r--r--widgets/text/e-completion-test.c220
-rw-r--r--widgets/text/e-completion-view.c94
-rw-r--r--widgets/text/e-completion-view.h4
-rw-r--r--widgets/text/e-completion.c363
-rw-r--r--widgets/text/e-completion.h21
-rw-r--r--widgets/text/e-entry.c78
-rw-r--r--widgets/text/e-text.c6
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;
}