ignore test-completion.

2003-02-09  Chris Toshok  <toshok@ximian.com>

	* tests/.cvsignore: ignore test-completion.

	* tests/test-completion.c: new program, completion test.

	* tests/Makefile.am (noinst_PROGRAMS): add test-completion.

	* gal/e-text/Makefile.am: remove e-completion-test from the build
	here, moving it to ../../tests.

	* gal/e-text/e-completion.c (e_completion_class_init): we've
	removed the restart, cancel, clear, and lost signals.  Also, we've
	renamed some so it's easier to tell from the name which it is
	(virtual func or signal.)
	(e_completion_dispose): remove call to clear_search_stack, as we
	don't do auto-refinement anymore.
	(e_completion_clear): gone.
	(e_completion_push_search): gone.
	(e_completion_pop_search): gone.
	(e_completion_clear_search_stack): gone.
	(e_completion_refine_search): gone.
	(e_completion_unrefine_search): gone.
	(e_completion_begin_search): substantially clear this up, since we
	don't have the refinement stuff anymore.  Also, the call to
	request_completion is a virtual function call, not a signal.
	(e_completion_match_count): always return matches->len here, never
	match_count, which is gone (with the refinement stuff)
	(e_completion_foreach_match): remove the hit_count stuff.
	(e_completion_restart): gone.
	(e_completion_lost_match): gone.
	(e_completion_end_search): remove the sorting stuff from here (and
	the call to restart.)  the etable sorting stuff will have to take
	up the slack, but for now there's no reason to restart the search
	here.

	* gal/e-text/e-completion.h (struct _ECompletionClass): straighten
	out what's a virtual function and what's a signal, instead of
	using signals for both.  Also, remove the auto_refine stuff, as
	it's not used.

	* gal/e-text/e-completion-view.c (e_completion_view_size_request):
	make the damn drop down window bigger (100 pixels, or the
	requisition height, whichever is bigger.)
	(e_completion_view_disconnect): remove handling for signals that
	are gone.
	(restart_completion_cb): gone.
	(cancel_completion_cb): gone.
	(clear_completion_cb): gone.
	(lost_completion_cb): gone.
	(e_completion_view_construct): track new names of ECompletion
	signals.

	* gal/e-text/e-completion-view.h (struct _ECompletionView): remove
	restart_signal_id, cancel_signal_id, clear_signal_id, and
	lost_signal_id.

	* gal/e-text/e-entry.c (get_borders): new function, ala gtkentry.
	(canvas_size_request): use get_borders instead of computing it
	here.
	(e_entry_init): remove duplicate assignment of
	emulate_label_resize.
	(e_entry_show_popup): remove some ifdef'ed crap.
	(e_entry_start_completion): don't cancel the completion before
	starting again.  This keeps the popup from disappearing.

	* gal/e-text/e-completion-match.c (e_completion_match_construct):
	no more hit_count.

	* gal/e-text/e-completion-match.h (struct _ECompletionMatch):
	remove hit_count.

	* gal/e-text/e-completion-callbacks.[ch]: new class so we can use
	callbacks instead of subclassing.

svn path=/trunk/; revision=19859
This commit is contained in:
Chris Toshok
2003-02-10 02:21:47 +00:00
committed by Chris Toshok
parent 23c65aa0bf
commit 18d96c5ff3
11 changed files with 240 additions and 714 deletions

View File

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

View File

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

View File

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

View File

@ -40,7 +40,6 @@ struct _ECompletionMatch {
gpointer user_data;
gint ref;
gint hit_count;
void (*destroy) (ECompletionMatch *);
};

View File

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

View File

@ -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 *
@ -593,28 +583,6 @@ begin_completion_cb (ECompletion *completion, const gchar *txt, gint pos, gint l
e_table_model_changed (cv->model);
}
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)
{
@ -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,

View File

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

View File

@ -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",
e_completion_signals[COMPLETION_STARTED] =
g_signal_new ("completion_started",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (ECompletionClass, request_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_BEGIN_COMPLETION] =
g_signal_new ("begin_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, begin_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_COMPLETION] =
g_signal_new ("completion",
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",
e_completion_signals[COMPLETION_FINISHED] =
g_signal_new ("completion_finished",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (ECompletionClass, restart_completion),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (ECompletionClass, completion_finished),
NULL, NULL,
e_marshal_NONE__NONE,
G_TYPE_NONE, 0);
e_completion_signals[E_COMPLETION_CANCEL_COMPLETION] =
g_signal_new ("cancel_completion",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (ECompletionClass, cancel_completion),
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;
@ -240,150 +178,10 @@ e_completion_clear_matches (ECompletion *complete)
complete->priv->max_score = 0;
}
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);
g_return_if_fail (comp != NULL);
g_return_if_fail (E_IS_COMPLETION (comp));
g_return_if_fail (comp->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);
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);
g_signal_emit (complete, e_completion_signals[E_COMPLETION_END_COMPLETION], 0);
complete->priv->searching = FALSE;
complete->priv->done_search = TRUE;
comp->priv->searching = FALSE;
comp->priv->done_search = TRUE;
}

View File

@ -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;
void (*request_completion) (ECompletion *comp, const gchar *search_text, gint pos, gint limit);
void (*end_completion) (ECompletion *comp);
/* Signals */
void (*request_completion) (ECompletion *comp, const gchar *search_text, gint pos, gint limit);
void (*completion_started) (ECompletion *comp, const gchar *search_text, gint pos, gint limit);
void (*begin_completion) (ECompletion *comp, const gchar *search_text, gint pos, gint limit);
void (*restart_completion) (ECompletion *comp);
void (*completion_found) (ECompletion *comp, ECompletionMatch *match);
void (*completion) (ECompletion *comp, ECompletionMatch *match);
void (*lost_completion) (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

View File

@ -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)
@ -165,6 +158,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;

View File

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