Blame gtk/gtkentrybuffer.c

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