2001-10-26 Christopher James Lahey <clahey@ximian.com> * gal/e-paned/e-hpaned.c, gal/e-paned/e-hpaned.h, gal/e-paned/e-paned.c, gal/e-paned/e-paned.h, gal/e-paned/e-vpaned.c, gal/e-paned/e-vpaned.h, gal/e-text/e-completion-match.c, gal/e-text/e-completion-match.h, gal/e-text/e-completion-test.c, gal/e-text/e-completion-view.c, gal/e-text/e-completion-view.h, gal/e-text/e-completion.c, gal/e-text/e-completion.h, gal/e-text/e-entry-test.c, gal/e-text/e-entry.c, gal/e-text/e-entry.h, gal/e-text/e-table-text-model.c, gal/e-text/e-table-text-model.h, gal/e-text/e-text-model-repos.c, gal/e-text/e-text-model-repos.h, gal/e-text/e-text-model-test.c, gal/e-text/e-text-model-uri.c, gal/e-text/e-text-model-uri.h, gal/e-text/e-text-model.c, gal/e-text/e-text-model.h, gal/e-text/e-text.c, gal/e-text/e-text.h, gal/util/e-bit-array.c, gal/util/e-bit-array.h, gal/util/e-cache.c, gal/util/e-cache.h, gal/util/e-iconv.c, gal/util/e-iconv.h, gal/util/e-sorter-array.c, gal/util/e-sorter-array.h, gal/util/e-sorter.c, gal/util/e-sorter.h, gal/util/e-text-event-processor-emacs-like.c, gal/util/e-text-event-processor-emacs-like.h, gal/util/e-text-event-processor-types.h, gal/util/e-text-event-processor.c, gal/util/e-text-event-processor.h, gal/util/e-util.c, gal/util/e-util.h, gal/util/e-xml-utils.c, gal/util/e-xml-utils.h, gal/widgets/color-group.c, gal/widgets/color-group.h, gal/widgets/color-palette.c, gal/widgets/color-palette.h, gal/widgets/e-canvas-utils.c, gal/widgets/e-canvas-utils.h, gal/widgets/e-canvas-vbox.c, gal/widgets/e-canvas-vbox.h, gal/widgets/e-canvas.c, gal/widgets/e-canvas.h, gal/widgets/e-categories-master-list-array.c, gal/widgets/e-categories-master-list-array.h, gal/widgets/e-categories-master-list-combo.c, gal/widgets/e-categories-master-list-combo.h, gal/widgets/e-categories-master-list-dialog-model.c, gal/widgets/e-categories-master-list-dialog-model.h, gal/widgets/e-categories-master-list-dialog.c, gal/widgets/e-categories-master-list-dialog.h, gal/widgets/e-categories-master-list.c, gal/widgets/e-categories-master-list.h, gal/widgets/e-categories.c, gal/widgets/e-categories.h, gal/widgets/e-colors.c, gal/widgets/e-colors.h, gal/widgets/e-cursors.c, gal/widgets/e-cursors.h, gal/widgets/e-font.c, gal/widgets/e-font.h, gal/widgets/e-gui-utils.c, gal/widgets/e-gui-utils.h, gal/widgets/e-hscrollbar.c, gal/widgets/e-hscrollbar.h, gal/widgets/e-popup-menu.c, gal/widgets/e-popup-menu.h, gal/widgets/e-printable.c, gal/widgets/e-printable.h, gal/widgets/e-reflow-model.c, gal/widgets/e-reflow-model.h, gal/widgets/e-reflow-sorted.c, gal/widgets/e-reflow-sorted.h, gal/widgets/e-reflow.c, gal/widgets/e-reflow.h, gal/widgets/e-scroll-frame.c, gal/widgets/e-scroll-frame.h, gal/widgets/e-selection-model-array.c, gal/widgets/e-selection-model-array.h, gal/widgets/e-selection-model-simple.c, gal/widgets/e-selection-model-simple.h, gal/widgets/e-selection-model.c, gal/widgets/e-selection-model.h, gal/widgets/e-unicode.c, gal/widgets/e-unicode.h, gal/widgets/e-vscrollbar.c, gal/widgets/e-vscrollbar.h, gal/widgets/gtk-combo-box.c, gal/widgets/gtk-combo-box.h, gal/widgets/gtk-combo-stack.c, gal/widgets/gtk-combo-stack.h, gal/widgets/gtk-combo-text.c, gal/widgets/gtk-combo-text.h, gal/widgets/test-color.c, gal/widgets/test-e-font.c, gal/widgets/test-e-font.h, gal/widgets/test-font-loading.c, gal/widgets/widget-color-combo.c, gal/widgets/widget-color-combo.h, gal/widgets/widget-pixmap-combo.c, gal/widgets/widget-pixmap-combo.h, src/e-table/e-table-sorted-variable.c, tests/test-define-views.c, tests/test-shortcut-bar.c, tests/test-table-1.c, tests/test-tree-1.c, tests/test-tree-2.c, tests/test-tree-3.c, tests/test-unicode.c: Changed the license announcement at the top of these files. svn path=/trunk/; revision=14158
646 lines
16 KiB
C
646 lines
16 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
|
||
/*
|
||
* e-text-model.c
|
||
* Copyright 2000, 2001, Ximian, Inc.
|
||
*
|
||
* Authors:
|
||
* Chris Lahey <clahey@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.
|
||
*/
|
||
|
||
#undef PARANOID_DEBUGGING
|
||
|
||
#include <config.h>
|
||
#include <ctype.h>
|
||
#include <string.h>
|
||
#include <gtk/gtksignal.h>
|
||
#include "e-text-model-repos.h"
|
||
#include "e-text-model.h"
|
||
#include "gal/util/e-util.h"
|
||
|
||
#define CLASS(obj) (E_TEXT_MODEL_CLASS (GTK_OBJECT (obj)->klass))
|
||
|
||
#define MAX_LENGTH (2047)
|
||
|
||
enum {
|
||
E_TEXT_MODEL_CHANGED,
|
||
E_TEXT_MODEL_REPOSITION,
|
||
E_TEXT_MODEL_OBJECT_ACTIVATED,
|
||
E_TEXT_MODEL_CANCEL_COMPLETION,
|
||
E_TEXT_MODEL_LAST_SIGNAL
|
||
};
|
||
|
||
static guint e_text_model_signals[E_TEXT_MODEL_LAST_SIGNAL] = { 0 };
|
||
|
||
struct _ETextModelPrivate {
|
||
gchar *text;
|
||
gint len;
|
||
};
|
||
|
||
static void e_text_model_class_init (ETextModelClass *class);
|
||
static void e_text_model_init (ETextModel *model);
|
||
static void e_text_model_destroy (GtkObject *object);
|
||
|
||
static gint e_text_model_real_validate_position (ETextModel *, gint pos);
|
||
static const gchar *e_text_model_real_get_text (ETextModel *model);
|
||
static gint e_text_model_real_get_text_length (ETextModel *model);
|
||
static void e_text_model_real_set_text (ETextModel *model, const gchar *text);
|
||
static void e_text_model_real_insert (ETextModel *model, gint postion, const gchar *text);
|
||
static void e_text_model_real_insert_length (ETextModel *model, gint postion, const gchar *text, gint length);
|
||
static void e_text_model_real_delete (ETextModel *model, gint postion, gint length);
|
||
|
||
static GtkObject *parent_class;
|
||
|
||
|
||
|
||
/**
|
||
* e_text_model_get_type:
|
||
* @void:
|
||
*
|
||
* Registers the &ETextModel class if necessary, and returns the type ID
|
||
* associated to it.
|
||
*
|
||
* Return value: The type ID of the &ETextModel class.
|
||
**/
|
||
GtkType
|
||
e_text_model_get_type (void)
|
||
{
|
||
static GtkType model_type = 0;
|
||
|
||
if (!model_type) {
|
||
GtkTypeInfo model_info = {
|
||
"ETextModel",
|
||
sizeof (ETextModel),
|
||
sizeof (ETextModelClass),
|
||
(GtkClassInitFunc) e_text_model_class_init,
|
||
(GtkObjectInitFunc) e_text_model_init,
|
||
NULL, /* reserved_1 */
|
||
NULL, /* reserved_2 */
|
||
(GtkClassInitFunc) NULL
|
||
};
|
||
|
||
model_type = gtk_type_unique (gtk_object_get_type (), &model_info);
|
||
}
|
||
|
||
return model_type;
|
||
}
|
||
|
||
/* Class initialization function for the text item */
|
||
static void
|
||
e_text_model_class_init (ETextModelClass *klass)
|
||
{
|
||
GtkObjectClass *object_class;
|
||
|
||
object_class = (GtkObjectClass *) klass;
|
||
|
||
parent_class = gtk_type_class (gtk_object_get_type ());
|
||
|
||
e_text_model_signals[E_TEXT_MODEL_CHANGED] =
|
||
gtk_signal_new ("changed",
|
||
GTK_RUN_LAST,
|
||
E_OBJECT_CLASS_TYPE (object_class),
|
||
GTK_SIGNAL_OFFSET (ETextModelClass, changed),
|
||
gtk_marshal_NONE__NONE,
|
||
GTK_TYPE_NONE, 0);
|
||
|
||
e_text_model_signals[E_TEXT_MODEL_REPOSITION] =
|
||
gtk_signal_new ("reposition",
|
||
GTK_RUN_LAST,
|
||
E_OBJECT_CLASS_TYPE (object_class),
|
||
GTK_SIGNAL_OFFSET (ETextModelClass, reposition),
|
||
gtk_marshal_NONE__POINTER_POINTER,
|
||
GTK_TYPE_NONE, 2,
|
||
GTK_TYPE_POINTER, GTK_TYPE_POINTER);
|
||
|
||
e_text_model_signals[E_TEXT_MODEL_OBJECT_ACTIVATED] =
|
||
gtk_signal_new ("object_activated",
|
||
GTK_RUN_LAST,
|
||
E_OBJECT_CLASS_TYPE (object_class),
|
||
GTK_SIGNAL_OFFSET (ETextModelClass, object_activated),
|
||
gtk_marshal_NONE__INT,
|
||
GTK_TYPE_NONE, 1,
|
||
GTK_TYPE_INT);
|
||
|
||
e_text_model_signals[E_TEXT_MODEL_CANCEL_COMPLETION] =
|
||
gtk_signal_new ("cancel_completion",
|
||
GTK_RUN_LAST,
|
||
E_OBJECT_CLASS_TYPE (object_class),
|
||
GTK_SIGNAL_OFFSET (ETextModelClass, cancel_completion),
|
||
gtk_marshal_NONE__NONE,
|
||
GTK_TYPE_NONE, 0);
|
||
|
||
E_OBJECT_CLASS_ADD_SIGNALS (object_class, e_text_model_signals, E_TEXT_MODEL_LAST_SIGNAL);
|
||
|
||
/* No default signal handlers. */
|
||
klass->changed = NULL;
|
||
klass->reposition = NULL;
|
||
klass->object_activated = NULL;
|
||
|
||
klass->validate_pos = e_text_model_real_validate_position;
|
||
|
||
klass->get_text = e_text_model_real_get_text;
|
||
klass->get_text_len = e_text_model_real_get_text_length;
|
||
klass->set_text = e_text_model_real_set_text;
|
||
klass->insert = e_text_model_real_insert;
|
||
klass->insert_length = e_text_model_real_insert_length;
|
||
klass->delete = e_text_model_real_delete;
|
||
|
||
/* We explicitly don't define default handlers for these. */
|
||
klass->objectify = NULL;
|
||
klass->obj_count = NULL;
|
||
klass->get_nth_obj = NULL;
|
||
|
||
object_class->destroy = e_text_model_destroy;
|
||
}
|
||
|
||
/* Object initialization function for the text item */
|
||
static void
|
||
e_text_model_init (ETextModel *model)
|
||
{
|
||
model->priv = g_new0 (struct _ETextModelPrivate, 1);
|
||
model->priv->text = g_strdup ("");
|
||
model->priv->len = 0;
|
||
}
|
||
|
||
/* Destroy handler for the text item */
|
||
static void
|
||
e_text_model_destroy (GtkObject *object)
|
||
{
|
||
ETextModel *model;
|
||
|
||
g_return_if_fail (object != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (object));
|
||
|
||
model = E_TEXT_MODEL (object);
|
||
|
||
g_free (model->priv->text);
|
||
|
||
g_free (model->priv);
|
||
model->priv = NULL;
|
||
|
||
if (GTK_OBJECT_CLASS (parent_class)->destroy)
|
||
GTK_OBJECT_CLASS (parent_class)->destroy (object);
|
||
}
|
||
|
||
static gint
|
||
e_text_model_real_validate_position (ETextModel *model, gint pos)
|
||
{
|
||
gint len;
|
||
|
||
if (pos < 0)
|
||
pos = 0;
|
||
else if (pos > ( len = e_text_model_get_text_length (model) ))
|
||
pos = len;
|
||
|
||
return pos;
|
||
}
|
||
|
||
static const gchar *
|
||
e_text_model_real_get_text (ETextModel *model)
|
||
{
|
||
if (model->priv->text)
|
||
return model->priv->text;
|
||
else
|
||
return "";
|
||
}
|
||
|
||
static gint
|
||
e_text_model_real_get_text_length (ETextModel *model)
|
||
{
|
||
if (model->priv->len < 0)
|
||
model->priv->len = strlen (e_text_model_get_text (model));
|
||
|
||
return model->priv->len;
|
||
}
|
||
|
||
static void
|
||
e_text_model_real_set_text (ETextModel *model, const gchar *text)
|
||
{
|
||
EReposAbsolute repos;
|
||
gboolean changed = FALSE;
|
||
|
||
if (text == NULL) {
|
||
|
||
changed = (model->priv->text != NULL);
|
||
|
||
g_free (model->priv->text);
|
||
model->priv->text = NULL;
|
||
model->priv->len = -1;
|
||
|
||
} else if (model->priv->text == NULL || strcmp (model->priv->text, text)) {
|
||
|
||
g_free (model->priv->text);
|
||
model->priv->text = g_strndup (text, MAX_LENGTH);
|
||
model->priv->len = -1;
|
||
|
||
changed = TRUE;
|
||
}
|
||
|
||
if (changed) {
|
||
e_text_model_changed (model);
|
||
repos.model = model;
|
||
repos.pos = -1;
|
||
e_text_model_reposition (model, e_repos_absolute, &repos);
|
||
}
|
||
}
|
||
|
||
static void
|
||
e_text_model_real_insert (ETextModel *model, gint position, const gchar *text)
|
||
{
|
||
EReposInsertShift repos;
|
||
gchar *new_text;
|
||
gint length;
|
||
|
||
if (model->priv->len < 0)
|
||
e_text_model_real_get_text_length (model);
|
||
length = strlen(text);
|
||
|
||
if (length + model->priv->len > MAX_LENGTH)
|
||
length = MAX_LENGTH - model->priv->len;
|
||
if (length <= 0)
|
||
return;
|
||
|
||
/* Can't use g_strdup_printf here because on some systems
|
||
printf ("%.*s"); is locale dependent. */
|
||
new_text = e_strdup_append_strings (model->priv->text, position,
|
||
text, length,
|
||
model->priv->text + position, -1,
|
||
NULL);
|
||
|
||
if (model->priv->text)
|
||
g_free (model->priv->text);
|
||
|
||
model->priv->text = new_text;
|
||
|
||
if (model->priv->len >= 0)
|
||
model->priv->len += length;
|
||
|
||
e_text_model_changed (model);
|
||
|
||
repos.model = model;
|
||
repos.pos = position;
|
||
repos.len = length;
|
||
|
||
e_text_model_reposition (model, e_repos_insert_shift, &repos);
|
||
}
|
||
|
||
static void
|
||
e_text_model_real_insert_length (ETextModel *model, gint position, const gchar *text, gint length)
|
||
{
|
||
EReposInsertShift repos;
|
||
gchar *new_text;
|
||
|
||
if (model->priv->len < 0)
|
||
e_text_model_real_get_text_length (model);
|
||
|
||
if (length + model->priv->len > MAX_LENGTH)
|
||
length = MAX_LENGTH - model->priv->len;
|
||
if (length <= 0)
|
||
return;
|
||
|
||
/* Can't use g_strdup_printf here because on some systems
|
||
printf ("%.*s"); is locale dependent. */
|
||
new_text = e_strdup_append_strings (model->priv->text, position,
|
||
text, length,
|
||
model->priv->text + position, -1,
|
||
NULL);
|
||
|
||
if (model->priv->text)
|
||
g_free (model->priv->text);
|
||
model->priv->text = new_text;
|
||
|
||
if (model->priv->len >= 0)
|
||
model->priv->len += length;
|
||
|
||
e_text_model_changed (model);
|
||
|
||
repos.model = model;
|
||
repos.pos = position;
|
||
repos.len = length;
|
||
|
||
e_text_model_reposition (model, e_repos_insert_shift, &repos);
|
||
}
|
||
|
||
static void
|
||
e_text_model_real_delete (ETextModel *model, gint position, gint length)
|
||
{
|
||
EReposDeleteShift repos;
|
||
|
||
memmove (model->priv->text + position, model->priv->text + position + length, strlen (model->priv->text + position + length) + 1);
|
||
|
||
if (model->priv->len >= 0)
|
||
model->priv->len -= length;
|
||
|
||
e_text_model_changed (model);
|
||
|
||
repos.model = model;
|
||
repos.pos = position;
|
||
repos.len = length;
|
||
|
||
e_text_model_reposition (model, e_repos_delete_shift, &repos);
|
||
}
|
||
|
||
void
|
||
e_text_model_changed (ETextModel *model)
|
||
{
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
|
||
/*
|
||
Objectify before emitting any signal.
|
||
While this method could, in theory, do pretty much anything, it is meant
|
||
for scanning objects and converting substrings into embedded objects.
|
||
*/
|
||
if (CLASS (model)->objectify)
|
||
CLASS (model)->objectify (model);
|
||
|
||
gtk_signal_emit (GTK_OBJECT (model),
|
||
e_text_model_signals[E_TEXT_MODEL_CHANGED]);
|
||
}
|
||
|
||
void
|
||
e_text_model_cancel_completion (ETextModel *model)
|
||
{
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
|
||
gtk_signal_emit (GTK_OBJECT (model), e_text_model_signals[E_TEXT_MODEL_CANCEL_COMPLETION]);
|
||
}
|
||
|
||
void
|
||
e_text_model_reposition (ETextModel *model, ETextModelReposFn fn, gpointer repos_data)
|
||
{
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
g_return_if_fail (fn != NULL);
|
||
|
||
gtk_signal_emit (GTK_OBJECT (model),
|
||
e_text_model_signals[E_TEXT_MODEL_REPOSITION],
|
||
fn, repos_data);
|
||
}
|
||
|
||
gint
|
||
e_text_model_validate_position (ETextModel *model, gint pos)
|
||
{
|
||
g_return_val_if_fail (model != NULL, 0);
|
||
g_return_val_if_fail (E_IS_TEXT_MODEL (model), 0);
|
||
|
||
if (CLASS (model)->validate_pos)
|
||
pos = CLASS (model)->validate_pos (model, pos);
|
||
|
||
return pos;
|
||
}
|
||
|
||
const gchar *
|
||
e_text_model_get_text (ETextModel *model)
|
||
{
|
||
g_return_val_if_fail (model != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_TEXT_MODEL (model), NULL);
|
||
|
||
if (CLASS (model)->get_text)
|
||
return CLASS (model)->get_text (model);
|
||
|
||
return "";
|
||
}
|
||
|
||
gint
|
||
e_text_model_get_text_length (ETextModel *model)
|
||
{
|
||
g_return_val_if_fail (model != NULL, 0);
|
||
g_return_val_if_fail (E_IS_TEXT_MODEL (model), 0);
|
||
|
||
if (CLASS (model)->get_text_len (model)) {
|
||
|
||
gint len = CLASS (model)->get_text_len (model);
|
||
|
||
#ifdef PARANOID_DEBUGGING
|
||
const gchar *str = e_text_model_get_text (model);
|
||
gint len2 = str ? strlen (str) : 0;
|
||
if (len != len)
|
||
g_error ("\"%s\" length reported as %d, not %d.", str, len, len2);
|
||
#endif
|
||
|
||
return len;
|
||
|
||
} else {
|
||
/* Calculate length the old-fashioned way... */
|
||
const gchar *str = e_text_model_get_text (model);
|
||
return str ? strlen (str) : 0;
|
||
}
|
||
}
|
||
|
||
void
|
||
e_text_model_set_text (ETextModel *model, const gchar *text)
|
||
{
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
|
||
if (CLASS (model)->set_text)
|
||
CLASS (model)->set_text (model, text);
|
||
}
|
||
|
||
void
|
||
e_text_model_insert (ETextModel *model, gint position, const gchar *text)
|
||
{
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
|
||
if (text == NULL)
|
||
return;
|
||
|
||
if (CLASS (model)->insert)
|
||
CLASS (model)->insert (model, position, text);
|
||
}
|
||
|
||
void
|
||
e_text_model_insert_length (ETextModel *model, gint position, const gchar *text, gint length)
|
||
{
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
g_return_if_fail (length >= 0);
|
||
|
||
|
||
if (text == NULL || length == 0)
|
||
return;
|
||
|
||
if (CLASS (model)->insert_length)
|
||
CLASS (model)->insert_length (model, position, text, length);
|
||
}
|
||
|
||
void
|
||
e_text_model_prepend (ETextModel *model, const gchar *text)
|
||
{
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
|
||
if (text == NULL)
|
||
return;
|
||
|
||
e_text_model_insert (model, 0, text);
|
||
}
|
||
|
||
void
|
||
e_text_model_append (ETextModel *model, const gchar *text)
|
||
{
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
|
||
if (text == NULL)
|
||
return;
|
||
|
||
e_text_model_insert (model, e_text_model_get_text_length (model), text);
|
||
}
|
||
|
||
void
|
||
e_text_model_delete (ETextModel *model, gint position, gint length)
|
||
{
|
||
gint txt_len;
|
||
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
g_return_if_fail (length >= 0);
|
||
|
||
txt_len = e_text_model_get_text_length (model);
|
||
if (position + length > txt_len)
|
||
length = txt_len - position;
|
||
|
||
if (length <= 0)
|
||
return;
|
||
|
||
if (CLASS (model)->delete)
|
||
CLASS (model)->delete (model, position, length);
|
||
}
|
||
|
||
gint
|
||
e_text_model_object_count (ETextModel *model)
|
||
{
|
||
g_return_val_if_fail (model != NULL, 0);
|
||
g_return_val_if_fail (E_IS_TEXT_MODEL (model), 0);
|
||
|
||
if (CLASS (model)->obj_count)
|
||
return CLASS (model)->obj_count (model);
|
||
|
||
return 0;
|
||
}
|
||
|
||
const gchar *
|
||
e_text_model_get_nth_object (ETextModel *model, gint n, gint *len)
|
||
{
|
||
g_return_val_if_fail (model != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_TEXT_MODEL (model), NULL);
|
||
|
||
if (n < 0 || n >= e_text_model_object_count (model))
|
||
return NULL;
|
||
|
||
if (CLASS (model)->get_nth_obj)
|
||
return CLASS (model)->get_nth_obj (model, n, len);
|
||
|
||
return NULL;
|
||
}
|
||
|
||
gchar *
|
||
e_text_model_strdup_nth_object (ETextModel *model, gint n)
|
||
{
|
||
const gchar *obj;
|
||
gint len = 0;
|
||
|
||
g_return_val_if_fail (model != NULL, NULL);
|
||
g_return_val_if_fail (E_IS_TEXT_MODEL (model), NULL);
|
||
|
||
obj = e_text_model_get_nth_object (model, n, &len);
|
||
|
||
return obj ? g_strndup (obj, n) : NULL;
|
||
}
|
||
|
||
void
|
||
e_text_model_get_nth_object_bounds (ETextModel *model, gint n, gint *start, gint *end)
|
||
{
|
||
const gchar *txt = NULL, *obj = NULL;
|
||
gint len = 0;
|
||
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
|
||
txt = e_text_model_get_text (model);
|
||
obj = e_text_model_get_nth_object (model, n, &len);
|
||
|
||
g_return_if_fail (obj != NULL);
|
||
|
||
if (start)
|
||
*start = obj - txt;
|
||
if (end)
|
||
*end = obj - txt + len;
|
||
}
|
||
|
||
gint
|
||
e_text_model_get_object_at_offset (ETextModel *model, gint offset)
|
||
{
|
||
g_return_val_if_fail (model != NULL, -1);
|
||
g_return_val_if_fail (E_IS_TEXT_MODEL (model), -1);
|
||
|
||
if (offset < 0 || offset >= e_text_model_get_text_length (model))
|
||
return -1;
|
||
|
||
/* If an optimized version has been provided, we use it. */
|
||
if (CLASS (model)->obj_at_offset) {
|
||
|
||
return CLASS (model)->obj_at_offset (model, offset);
|
||
|
||
} else {
|
||
/* If not, we fake it.*/
|
||
|
||
gint i, N, pos0, pos1;
|
||
|
||
N = e_text_model_object_count (model);
|
||
|
||
for (i = 0; i < N; ++i) {
|
||
e_text_model_get_nth_object_bounds (model, i, &pos0, &pos1);
|
||
if (pos0 <= offset && offset < pos1)
|
||
return i;
|
||
}
|
||
|
||
}
|
||
|
||
return -1;
|
||
}
|
||
|
||
gint
|
||
e_text_model_get_object_at_pointer (ETextModel *model, const gchar *s)
|
||
{
|
||
g_return_val_if_fail (model != NULL, -1);
|
||
g_return_val_if_fail (E_IS_TEXT_MODEL (model), -1);
|
||
g_return_val_if_fail (s != NULL, -1);
|
||
|
||
return e_text_model_get_object_at_offset (model, s - e_text_model_get_text (model));
|
||
}
|
||
|
||
void
|
||
e_text_model_activate_nth_object (ETextModel *model, gint n)
|
||
{
|
||
g_return_if_fail (model != NULL);
|
||
g_return_if_fail (E_IS_TEXT_MODEL (model));
|
||
g_return_if_fail (n >= 0);
|
||
g_return_if_fail (n < e_text_model_object_count (model));
|
||
|
||
gtk_signal_emit (GTK_OBJECT (model), e_text_model_signals[E_TEXT_MODEL_OBJECT_ACTIVATED], n);
|
||
}
|
||
|
||
ETextModel *
|
||
e_text_model_new (void)
|
||
{
|
||
ETextModel *model = gtk_type_new (e_text_model_get_type ());
|
||
return model;
|
||
}
|