summaryrefslogtreecommitdiff
path: root/gtk/gtkentrybuffer.c
diff options
context:
space:
mode:
Diffstat (limited to 'gtk/gtkentrybuffer.c')
-rw-r--r--gtk/gtkentrybuffer.c741
1 files changed, 741 insertions, 0 deletions
diff --git a/gtk/gtkentrybuffer.c b/gtk/gtkentrybuffer.c
new file mode 100644
index 000000000..7c6d2186a
--- /dev/null
+++ b/gtk/gtkentrybuffer.c
@@ -0,0 +1,741 @@
+/* gtkentrybuffer.c
+ * Copyright (C) 2009 Stefan Walter <stef@memberwebs.com>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * 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 "gtkentrybuffer.h"
+#include "gtkintl.h"
+#include "gtkmarshalers.h"
+#include "gtkprivate.h"
+#include "gtkwidget.h"
+
+#include <gdk/gdk.h>
+
+#include <string.h>
+
+/**
+ * SECTION:gtkentrybuffer
+ * @title: GtkEntryBuffer
+ * @short_description: Text buffer for GtkEntry
+ *
+ * The #GtkEntryBuffer class contains the actual text displayed in a
+ * #GtkEntry widget.
+ *
+ * A single #GtkEntryBuffer object can be shared by multiple #GtkEntry
+ * widgets which will then share the same text content, but not the cursor
+ * position, visibility attributes, icon etc.
+ *
+ * #GtkEntryBuffer may be derived from. Such a derived class might allow
+ * text to be stored in an alternate location, such as non-pageable memory,
+ * useful in the case of important passwords. Or a derived class could
+ * integrate with an application's concept of undo/redo.
+ *
+ * Since: 2.18
+ */
+
+/* Initial size of buffer, in bytes */
+#define MIN_SIZE 16
+
+enum {
+ PROP_0,
+ PROP_TEXT,
+ PROP_LENGTH,
+ PROP_MAX_LENGTH,
+};
+
+enum {
+ INSERTED_TEXT,
+ DELETED_TEXT,
+ LAST_SIGNAL
+};
+
+static guint signals[LAST_SIGNAL] = { 0 };
+
+struct _GtkEntryBufferPrivate
+{
+ guint max_length;
+
+ /* Only valid if this class is not derived */
+ gchar *normal_text;
+ gsize normal_text_size;
+ gsize normal_text_bytes;
+ guint normal_text_chars;
+};
+
+G_DEFINE_TYPE (GtkEntryBuffer, gtk_entry_buffer, G_TYPE_OBJECT);
+
+/* --------------------------------------------------------------------------------
+ * DEFAULT IMPLEMENTATIONS OF TEXT BUFFER
+ *
+ * These may be overridden by a derived class, behavior may be changed etc...
+ * The normal_text and normal_text_xxxx fields may not be valid when
+ * this class is derived from.
+ */
+
+/* Overwrite a memory that might contain sensitive information. */
+static void
+trash_area (gchar *area,
+ gsize len)
+{
+ volatile gchar *varea = (volatile gchar *)area;
+ while (len-- > 0)
+ *varea++ = 0;
+}
+
+static const gchar*
+gtk_entry_buffer_normal_get_text (GtkEntryBuffer *buffer,
+ gsize *n_bytes)
+{
+ if (n_bytes)
+ *n_bytes = buffer->priv->normal_text_bytes;
+ if (!buffer->priv->normal_text)
+ return "";
+ return buffer->priv->normal_text;
+}
+
+static guint
+gtk_entry_buffer_normal_get_length (GtkEntryBuffer *buffer)
+{
+ return buffer->priv->normal_text_chars;
+}
+
+static guint
+gtk_entry_buffer_normal_insert_text (GtkEntryBuffer *buffer,
+ guint position,
+ const gchar *chars,
+ guint n_chars)
+{
+ GtkEntryBufferPrivate *pv = buffer->priv;
+ gsize prev_size;
+ gsize n_bytes;
+ gsize at;
+
+ n_bytes = g_utf8_offset_to_pointer (chars, n_chars) - chars;
+
+ /* Need more memory */
+ if (n_bytes + pv->normal_text_bytes + 1 > pv->normal_text_size)
+ {
+ prev_size = pv->normal_text_size;
+
+ /* Calculate our new buffer size */
+ while (n_bytes + pv->normal_text_bytes + 1 > pv->normal_text_size)
+ {
+ if (pv->normal_text_size == 0)
+ pv->normal_text_size = MIN_SIZE;
+ else
+ {
+ if (2 * pv->normal_text_size < GTK_ENTRY_BUFFER_MAX_SIZE)
+ pv->normal_text_size *= 2;
+ else
+ {
+ pv->normal_text_size = GTK_ENTRY_BUFFER_MAX_SIZE;
+ if (n_bytes > pv->normal_text_size - pv->normal_text_bytes - 1)
+ {
+ n_bytes = pv->normal_text_size - pv->normal_text_bytes - 1;
+ n_bytes = g_utf8_find_prev_char (chars, chars + n_bytes + 1) - chars;
+ n_chars = g_utf8_strlen (chars, n_bytes);
+ }
+ break;
+ }
+ }
+ }
+
+ /* Could be a password, so can't leave stuff in memory. */
+ gchar *et_new = g_malloc (pv->normal_text_size);
+ memcpy (et_new, pv->normal_text, MIN (prev_size, pv->normal_text_size));
+ trash_area (pv->normal_text, prev_size);
+ g_free (pv->normal_text);
+ pv->normal_text = et_new;
+ }
+
+ /* Actual text insertion */
+ at = g_utf8_offset_to_pointer (pv->normal_text, position) - pv->normal_text;
+ g_memmove (pv->normal_text + at + n_bytes, pv->normal_text + at, pv->normal_text_bytes - at);
+ memcpy (pv->normal_text + at, chars, n_bytes);
+
+ /* Book keeping */
+ pv->normal_text_bytes += n_bytes;
+ pv->normal_text_chars += n_chars;
+ pv->normal_text[pv->normal_text_bytes] = '\0';
+
+ gtk_entry_buffer_emit_inserted_text (buffer, position, chars, n_chars);
+ return n_chars;
+}
+
+static guint
+gtk_entry_buffer_normal_delete_text (GtkEntryBuffer *buffer,
+ guint position,
+ guint n_chars)
+{
+ GtkEntryBufferPrivate *pv = buffer->priv;
+ gsize start, end;
+
+ if (position > pv->normal_text_chars)
+ position = pv->normal_text_chars;
+ if (position + n_chars > pv->normal_text_chars)
+ n_chars = pv->normal_text_chars - position;
+
+ if (n_chars > 0)
+ {
+ start = g_utf8_offset_to_pointer (pv->normal_text, position) - pv->normal_text;
+ end = g_utf8_offset_to_pointer (pv->normal_text, position + n_chars) - pv->normal_text;
+
+ g_memmove (pv->normal_text + start, pv->normal_text + end, pv->normal_text_bytes + 1 - end);
+ pv->normal_text_chars -= n_chars;
+ pv->normal_text_bytes -= (end - start);
+
+ /*
+ * Could be a password, make sure we don't leave anything sensitive after
+ * the terminating zero. Note, that the terminating zero already trashed
+ * one byte.
+ */
+ trash_area (pv->normal_text + pv->normal_text_bytes + 1, end - start - 1);
+
+ gtk_entry_buffer_emit_deleted_text (buffer, position, n_chars);
+ }
+
+ return n_chars;
+}
+
+/* --------------------------------------------------------------------------------
+ *
+ */
+
+static void
+gtk_entry_buffer_real_inserted_text (GtkEntryBuffer *buffer,
+ guint position,
+ const gchar *chars,
+ guint n_chars)
+{
+ g_object_notify (G_OBJECT (buffer), "text");
+ g_object_notify (G_OBJECT (buffer), "length");
+}
+
+static void
+gtk_entry_buffer_real_deleted_text (GtkEntryBuffer *buffer,
+ guint position,
+ guint n_chars)
+{
+ g_object_notify (G_OBJECT (buffer), "text");
+ g_object_notify (G_OBJECT (buffer), "length");
+}
+
+/* --------------------------------------------------------------------------------
+ *
+ */
+
+static void
+gtk_entry_buffer_init (GtkEntryBuffer *buffer)
+{
+ GtkEntryBufferPrivate *pv;
+
+ pv = buffer->priv = G_TYPE_INSTANCE_GET_PRIVATE (buffer, GTK_TYPE_ENTRY_BUFFER, GtkEntryBufferPrivate);
+
+ pv->normal_text = NULL;
+ pv->normal_text_chars = 0;
+ pv->normal_text_bytes = 0;
+ pv->normal_text_size = 0;
+}
+
+static void
+gtk_entry_buffer_finalize (GObject *obj)
+{
+ GtkEntryBuffer *buffer = GTK_ENTRY_BUFFER (obj);
+ GtkEntryBufferPrivate *pv = buffer->priv;
+
+ if (pv->normal_text)
+ {
+ trash_area (pv->normal_text, pv->normal_text_size);
+ g_free (pv->normal_text);
+ pv->normal_text = NULL;
+ pv->normal_text_bytes = pv->normal_text_size = 0;
+ pv->normal_text_chars = 0;
+ }
+
+ G_OBJECT_CLASS (gtk_entry_buffer_parent_class)->finalize (obj);
+}
+
+static void
+gtk_entry_buffer_set_property (GObject *obj,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ GtkEntryBuffer *buffer = GTK_ENTRY_BUFFER (obj);
+
+ switch (prop_id)
+ {
+ case PROP_TEXT:
+ gtk_entry_buffer_set_text (buffer, g_value_get_string (value), -1);
+ break;
+ case PROP_MAX_LENGTH:
+ gtk_entry_buffer_set_max_length (buffer, g_value_get_uint (value));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gtk_entry_buffer_get_property (GObject *obj,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ GtkEntryBuffer *buffer = GTK_ENTRY_BUFFER (obj);
+
+ switch (prop_id)
+ {
+ case PROP_TEXT:
+ g_value_set_string (value, gtk_entry_buffer_get_text (buffer));
+ break;
+ case PROP_LENGTH:
+ g_value_set_uint (value, gtk_entry_buffer_get_length (buffer));
+ break;
+ case PROP_MAX_LENGTH:
+ g_value_set_uint (value, gtk_entry_buffer_get_max_length (buffer));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gtk_entry_buffer_class_init (GtkEntryBufferClass *klass)
+{
+ GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
+
+ gobject_class->finalize = gtk_entry_buffer_finalize;
+ gobject_class->set_property = gtk_entry_buffer_set_property;
+ gobject_class->get_property = gtk_entry_buffer_get_property;
+
+ klass->get_text = gtk_entry_buffer_normal_get_text;
+ klass->get_length = gtk_entry_buffer_normal_get_length;
+ klass->insert_text = gtk_entry_buffer_normal_insert_text;
+ klass->delete_text = gtk_entry_buffer_normal_delete_text;
+
+ klass->inserted_text = gtk_entry_buffer_real_inserted_text;
+ klass->deleted_text = gtk_entry_buffer_real_deleted_text;
+
+ g_type_class_add_private (gobject_class, sizeof (GtkEntryBufferPrivate));
+
+ /**
+ * GtkEntryBuffer:text:
+ *
+ * The contents of the buffer.
+ *
+ * Since: 2.18
+ */
+ g_object_class_install_property (gobject_class, PROP_TEXT,
+ g_param_spec_string ("text", P_("Text"),
+ P_("The contents of the buffer"),
+ "", GTK_PARAM_READWRITE));
+
+ /**
+ * GtkEntryBuffer:length:
+ *
+ * The length (in characters) of the text in buffer.
+ *
+ * Since: 2.18
+ */
+ g_object_class_install_property (gobject_class, PROP_LENGTH,
+ g_param_spec_uint ("length", P_("Text length"),
+ P_("Length of the text currently in the buffer"),
+ 0, GTK_ENTRY_BUFFER_MAX_SIZE, 0, GTK_PARAM_READABLE));
+
+ /**
+ * GtkEntryBuffer:max-length:
+ *
+ * The maximum length (in characters) of the text in the buffer.
+ *
+ * Since: 2.18
+ */
+ g_object_class_install_property (gobject_class, PROP_MAX_LENGTH,
+ g_param_spec_uint ("max-length", P_("Maximum length"),
+ P_("Maximum number of characters for this entry. Zero if no maximum"),
+ 0, GTK_ENTRY_BUFFER_MAX_SIZE, 0, GTK_PARAM_READWRITE));
+
+ /**
+ * GtkEntry::inserted-text:
+ * @buffer: a #GtkEntryBuffer
+ * @position: the position the text was inserted at.
+ * @chars: The text that was inserted.
+ * @n_chars: The number of characters that were inserted.
+ *
+ * This signal is emitted after text is inserted into the buffer.
+ *
+ * Since: 2.18
+ */
+ signals[INSERTED_TEXT] = g_signal_new (I_("inserted-text"),
+ GTK_TYPE_ENTRY_BUFFER,
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (GtkEntryBufferClass, inserted_text),
+ NULL, NULL,
+ _gtk_marshal_VOID__UINT_STRING_UINT,
+ G_TYPE_NONE, 3,
+ G_TYPE_UINT,
+ G_TYPE_STRING,
+ G_TYPE_UINT);
+
+ /**
+ * GtkEntry::deleted-text:
+ * @buffer: a #GtkEntryBuffer
+ * @position: the position the text was deleted at.
+ * @n_chars: The number of characters that were deleted.
+ *
+ * This signal is emitted after text is deleted from the buffer.
+ *
+ * Since: 2.18
+ */
+ signals[DELETED_TEXT] = g_signal_new (I_("deleted-text"),
+ GTK_TYPE_ENTRY_BUFFER,
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (GtkEntryBufferClass, deleted_text),
+ NULL, NULL,
+ _gtk_marshal_VOID__UINT_UINT,
+ G_TYPE_NONE, 2,
+ G_TYPE_UINT,
+ G_TYPE_UINT);
+}
+
+/* --------------------------------------------------------------------------------
+ *
+ */
+
+/**
+ * gtk_entry_buffer_new:
+ * @initial_chars: initial buffer text, or %NULL
+ * @n_initial_chars: number of characters in @initial_chars, or -1
+ *
+ * Create a new GtkEntryBuffer object.
+ *
+ * Optionally, specify initial text to set in the buffer.
+ *
+ * Return value: A new GtkEntryBuffer object.
+ *
+ * Since: 2.18
+ **/
+GtkEntryBuffer*
+gtk_entry_buffer_new (const gchar *initial_chars,
+ gint n_initial_chars)
+{
+ GtkEntryBuffer *buffer = g_object_new (GTK_TYPE_ENTRY_BUFFER, NULL);
+ if (initial_chars)
+ gtk_entry_buffer_set_text (buffer, initial_chars, n_initial_chars);
+ return buffer;
+}
+
+/**
+ * gtk_entry_buffer_get_length:
+ * @buffer: a #GtkEntryBuffer
+ *
+ * Retrieves the length in characters of the buffer.
+ *
+ * Return value: The number of characters in the buffer.
+ *
+ * Since: 2.18
+ **/
+guint
+gtk_entry_buffer_get_length (GtkEntryBuffer *buffer)
+{
+ GtkEntryBufferClass *klass;
+
+ g_return_val_if_fail (GTK_IS_ENTRY_BUFFER (buffer), 0);
+
+ klass = GTK_ENTRY_BUFFER_GET_CLASS (buffer);
+ g_return_val_if_fail (klass->get_length != NULL, 0);
+
+ return (*klass->get_length) (buffer);
+}
+
+/**
+ * gtk_entry_buffer_get_bytes:
+ * @buffer: a #GtkEntryBuffer
+ *
+ * Retrieves the length in bytes of the buffer.
+ * See gtk_entry_buffer_get_length().
+ *
+ * Return value: The byte length of the buffer.
+ *
+ * Since: 2.18
+ **/
+gsize
+gtk_entry_buffer_get_bytes (GtkEntryBuffer *buffer)
+{
+ GtkEntryBufferClass *klass;
+ gsize bytes = 0;
+
+ g_return_val_if_fail (GTK_IS_ENTRY_BUFFER (buffer), 0);
+
+ klass = GTK_ENTRY_BUFFER_GET_CLASS (buffer);
+ g_return_val_if_fail (klass->get_text != NULL, 0);
+
+ (*klass->get_text) (buffer, &bytes);
+ return bytes;
+}
+
+/**
+ * gtk_entry_buffer_get_text:
+ * @buffer: a #GtkEntryBuffer
+ *
+ * Retrieves the contents of the buffer.
+ *
+ * The memory pointer returned by this call will not change
+ * unless this object emits a signal, or is finalized.
+ *
+ * Return value: a pointer to the contents of the widget as a
+ * string. This string points to internally allocated
+ * storage in the buffer and must not be freed, modified or
+ * stored.
+ *
+ * Since: 2.18
+ **/
+const gchar*
+gtk_entry_buffer_get_text (GtkEntryBuffer *buffer)
+{
+ GtkEntryBufferClass *klass;
+
+ g_return_val_if_fail (GTK_IS_ENTRY_BUFFER (buffer), NULL);
+
+ klass = GTK_ENTRY_BUFFER_GET_CLASS (buffer);
+ g_return_val_if_fail (klass->get_text != NULL, NULL);
+
+ return (*klass->get_text) (buffer, NULL);
+}
+
+/**
+ * gtk_entry_buffer_set_text:
+ * @buffer: a #GtkEntryBuffer
+ * @chars: the new text
+ * @n_chars: the number of characters in @text, or -1
+ *
+ * Sets the text in the buffer.
+ *
+ * This is roughly equivalent to calling gtk_entry_buffer_delete_text()
+ * and gtk_entry_buffer_insert_text().
+ *
+ * Note that @n_chars is in characters, not in bytes.
+ *
+ * Since: 2.18
+ **/
+void
+gtk_entry_buffer_set_text (GtkEntryBuffer *buffer,
+ const gchar *chars,
+ gint n_chars)
+{
+ g_return_if_fail (GTK_IS_ENTRY_BUFFER (buffer));
+ g_return_if_fail (chars != NULL);
+
+ g_object_freeze_notify (G_OBJECT (buffer));
+ gtk_entry_buffer_delete_text (buffer, 0, -1);
+ gtk_entry_buffer_insert_text (buffer, 0, chars, n_chars);
+ g_object_thaw_notify (G_OBJECT (buffer));
+}
+
+/**
+ * gtk_entry_buffer_set_max_length:
+ * @buffer: a #GtkEntryBuffer
+ * @max_length: the maximum length of the entry buffer, or 0 for no maximum.
+ * (other than the maximum length of entries.) The value passed in will
+ * be clamped to the range 0-65536.
+ *
+ * Sets the maximum allowed length of the contents of the buffer. If
+ * the current contents are longer than the given length, then they
+ * will be truncated to fit.
+ *
+ * Since: 2.18
+ **/
+void
+gtk_entry_buffer_set_max_length (GtkEntryBuffer *buffer,
+ guint max_length)
+{
+ g_return_if_fail (GTK_IS_ENTRY_BUFFER (buffer));
+
+ if (max_length > GTK_ENTRY_BUFFER_MAX_SIZE)
+ max_length = GTK_ENTRY_BUFFER_MAX_SIZE;
+
+ if (max_length > 0 && gtk_entry_buffer_get_length (buffer) > max_length)
+ gtk_entry_buffer_delete_text (buffer, max_length, -1);
+
+ buffer->priv->max_length = max_length;
+ g_object_notify (G_OBJECT (buffer), "max-length");
+}
+
+/**
+ * gtk_entry_buffer_get_max_length:
+ * @buffer: a #GtkEntryBuffer
+ *
+ * Retrieves the maximum allowed length of the text in
+ * @buffer. See gtk_entry_buffer_set_max_length().
+ *
+ * Return value: the maximum allowed number of characters
+ * in #GtkEntryBuffer, or 0 if there is no maximum.
+ *
+ * Since: 2.18
+ **/
+guint
+gtk_entry_buffer_get_max_length (GtkEntryBuffer *buffer)
+{
+ g_return_val_if_fail (GTK_IS_ENTRY_BUFFER (buffer), 0);
+ return buffer->priv->max_length;
+}
+
+/**
+ * gtk_entry_buffer_insert_text:
+ * @buffer: a #GtkEntryBuffer
+ * @position: the position at which to insert text.
+ * @chars: the text to insert into the buffer.
+ * @n_chars: the length of the text in characters, or -1
+ *
+ * Inserts @n_chars characters of @chars into the contents of the
+ * buffer, at position @position.
+ *
+ * If @n_chars is negative, then characters from chars will be inserted
+ * until a null-terminator is found. If @position or @n_chars are out of
+ * bounds, or the maximum buffer text length is exceeded, then they are
+ * coerced to sane values.
+ *
+ * Note that the position and length are in characters, not in bytes.
+ *
+ * Returns: The number of characters actually inserted.
+ *
+ * Since: 2.18
+ */
+guint
+gtk_entry_buffer_insert_text (GtkEntryBuffer *buffer,
+ guint position,
+ const gchar *chars,
+ gint n_chars)
+{
+ GtkEntryBufferClass *klass;
+ GtkEntryBufferPrivate *pv;
+ guint length;
+
+ g_return_val_if_fail (GTK_IS_ENTRY_BUFFER (buffer), 0);
+
+ length = gtk_entry_buffer_get_length (buffer);
+ pv = buffer->priv;
+
+ if (n_chars < 0)
+ n_chars = g_utf8_strlen (chars, -1);
+
+ /* Bring position into bounds */
+ if (position > length)
+ position = length;
+
+ /* Make sure not entering too much data */
+ if (pv->max_length > 0)
+ {
+ if (length >= pv->max_length)
+ n_chars = 0;
+ else if (length + n_chars > pv->max_length)
+ n_chars -= (length + n_chars) - pv->max_length;
+ }
+
+ klass = GTK_ENTRY_BUFFER_GET_CLASS (buffer);
+ g_return_val_if_fail (klass->insert_text != NULL, 0);
+
+ return (*klass->insert_text) (buffer, position, chars, n_chars);
+}
+
+/**
+ * gtk_entry_buffer_delete_text:
+ * @buffer: a #GtkEntryBuffer
+ * @position: position at which to delete text
+ * @n_chars: number of characters to delete
+ *
+ * Deletes a sequence of characters from the buffer. @n_chars characters are
+ * deleted starting at @position. If @n_chars is negative, then all characters
+ * until the end of the text are deleted.
+ *
+ * If @position or @n_chars are out of bounds, then they are coerced to sane
+ * values.
+ *
+ * Note that the positions are specified in characters, not bytes.
+ *
+ * Returns: The number of characters deleted.
+ *
+ * Since: 2.18
+ */
+guint
+gtk_entry_buffer_delete_text (GtkEntryBuffer *buffer,
+ guint position,
+ gint n_chars)
+{
+ GtkEntryBufferClass *klass;
+ guint length;
+
+ g_return_val_if_fail (GTK_IS_ENTRY_BUFFER (buffer), 0);
+
+ length = gtk_entry_buffer_get_length (buffer);
+ if (n_chars < 0)
+ n_chars = length;
+ if (position > length)
+ position = length;
+ if (position + n_chars > length)
+ n_chars = length - position;
+
+ klass = GTK_ENTRY_BUFFER_GET_CLASS (buffer);
+ g_return_val_if_fail (klass->delete_text != NULL, 0);
+
+ return (*klass->delete_text) (buffer, position, n_chars);
+}
+
+/**
+ * gtk_entry_buffer_emit_inserted_text:
+ * @buffer: a #GtkEntryBuffer
+ * @position: position at which text was inserted
+ * @chars: text that was inserted
+ * @n_chars: number of characters inserted
+ *
+ * Used when subclassing #GtkEntryBuffer
+ *
+ * Since: 2.18
+ */
+void
+gtk_entry_buffer_emit_inserted_text (GtkEntryBuffer *buffer,
+ guint position,
+ const gchar *chars,
+ guint n_chars)
+{
+ g_return_if_fail (GTK_IS_ENTRY_BUFFER (buffer));
+ g_signal_emit (buffer, signals[INSERTED_TEXT], 0, position, chars, n_chars);
+}
+
+/**
+ * gtk_entry_buffer_emit_deleted_text:
+ * @buffer: a #GtkEntryBuffer
+ * @position: position at which text was deleted
+ * @n_chars: number of characters deleted
+ *
+ * Used when subclassing #GtkEntryBuffer
+ *
+ * Since: 2.18
+ */
+void
+gtk_entry_buffer_emit_deleted_text (GtkEntryBuffer *buffer,
+ guint position,
+ guint n_chars)
+{
+ g_return_if_fail (GTK_IS_ENTRY_BUFFER (buffer));
+ g_signal_emit (buffer, signals[DELETED_TEXT], 0, position, n_chars);
+}
+