752 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			752 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* 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, see <http://www.gnu.org/licenses/>.
 | ||
|  */
 | ||
| 
 | ||
| #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,
 | ||
|   NUM_PROPERTIES
 | ||
| };
 | ||
| 
 | ||
| static GParamSpec *entry_buffer_props[NUM_PROPERTIES] = { NULL, };
 | ||
| 
 | ||
| enum {
 | ||
|   INSERTED_TEXT,
 | ||
|   DELETED_TEXT,
 | ||
|   LAST_SIGNAL
 | ||
| };
 | ||
| 
 | ||
| static guint signals[LAST_SIGNAL] = { 0 };
 | ||
| 
 | ||
| struct _GtkEntryBufferPrivate
 | ||
| {
 | ||
|   /* Only valid if this class is not derived */
 | ||
|   gchar *normal_text;
 | ||
|   gsize  normal_text_size;
 | ||
|   gsize  normal_text_bytes;
 | ||
|   guint  normal_text_chars;
 | ||
| 
 | ||
|   gint   max_length;
 | ||
| };
 | ||
| 
 | ||
| G_DEFINE_TYPE_WITH_PRIVATE (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)
 | ||
|     {
 | ||
|       gchar *et_new;
 | ||
| 
 | ||
|       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. */
 | ||
|       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;
 | ||
|   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;
 | ||
| 
 | ||
|       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_by_pspec (G_OBJECT (buffer), entry_buffer_props[PROP_TEXT]);
 | ||
|   g_object_notify_by_pspec (G_OBJECT (buffer), entry_buffer_props[PROP_LENGTH]);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| gtk_entry_buffer_real_deleted_text (GtkEntryBuffer *buffer,
 | ||
|                                     guint           position,
 | ||
|                                     guint           n_chars)
 | ||
| {
 | ||
|   g_object_notify_by_pspec (G_OBJECT (buffer), entry_buffer_props[PROP_TEXT]);
 | ||
|   g_object_notify_by_pspec (G_OBJECT (buffer), entry_buffer_props[PROP_LENGTH]);
 | ||
| }
 | ||
| 
 | ||
| /* --------------------------------------------------------------------------------
 | ||
|  *
 | ||
|  */
 | ||
| 
 | ||
| static void
 | ||
| gtk_entry_buffer_init (GtkEntryBuffer *buffer)
 | ||
| {
 | ||
|   GtkEntryBufferPrivate *pv;
 | ||
| 
 | ||
|   pv = buffer->priv = gtk_entry_buffer_get_instance_private (buffer);
 | ||
| 
 | ||
|   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_int (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_int (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;
 | ||
| 
 | ||
|   /**
 | ||
|    * GtkEntryBuffer:text:
 | ||
|    *
 | ||
|    * The contents of the buffer.
 | ||
|    *
 | ||
|    * Since: 2.18
 | ||
|    */
 | ||
|   entry_buffer_props[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
 | ||
|    */
 | ||
|    entry_buffer_props[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
 | ||
|    */
 | ||
|   entry_buffer_props[PROP_MAX_LENGTH] =
 | ||
|       g_param_spec_int ("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|G_PARAM_EXPLICIT_NOTIFY);
 | ||
| 
 | ||
|   g_object_class_install_properties (gobject_class, NUM_PROPERTIES, entry_buffer_props);
 | ||
| 
 | ||
|   /**
 | ||
|    * GtkEntryBuffer::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);
 | ||
| 
 | ||
|   /**
 | ||
|    * GtkEntryBuffer::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: (allow-none): 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.
 | ||
|  *
 | ||
|  * Returns: 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.
 | ||
|  *
 | ||
|  * Returns: 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().
 | ||
|  *
 | ||
|  * Returns: 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.
 | ||
|  *
 | ||
|  * Returns: 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,
 | ||
|                                  gint            max_length)
 | ||
| {
 | ||
|   g_return_if_fail (GTK_IS_ENTRY_BUFFER (buffer));
 | ||
| 
 | ||
|   max_length = CLAMP (max_length, 0, GTK_ENTRY_BUFFER_MAX_SIZE);
 | ||
| 
 | ||
|   if (buffer->priv->max_length == max_length)
 | ||
|     return;
 | ||
| 
 | ||
|   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_by_pspec (G_OBJECT (buffer), entry_buffer_props[PROP_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().
 | ||
|  *
 | ||
|  * Returns: the maximum allowed number of characters
 | ||
|  *               in #GtkEntryBuffer, or 0 if there is no maximum.
 | ||
|  *
 | ||
|  * Since: 2.18
 | ||
|  */
 | ||
| gint
 | ||
| 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);
 | ||
| }
 | 
