/* clutter-text-buffer.c * Copyright (C) 2011 Collabora Ltd. * * 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 . * * Author: Stef Walter */ #include "config.h" #include "clutter/clutter-text-buffer.h" #include "clutter/clutter-marshal.h" #include "clutter/clutter-private.h" #include /** * ClutterTextBuffer: * * Text buffer for [class@Text] * * The #ClutterTextBuffer class contains the actual text displayed in a * [class@Text] widget. * * A single #ClutterTextBuffer object can be shared by multiple [class@Text] * widgets which will then share the same text content, but not the cursor * position, visibility attributes, icon etc. * * #ClutterTextBuffer 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. */ /* Initial size of buffer, in bytes */ #define MIN_SIZE 16 enum { PROP_0, PROP_TEXT, PROP_LENGTH, PROP_MAX_LENGTH, PROP_LAST }; static GParamSpec *obj_props[PROP_LAST] = { NULL, }; enum { INSERTED_TEXT, DELETED_TEXT, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; typedef struct _ClutterTextBufferPrivate { gint 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; } ClutterTextBufferPrivate; G_DEFINE_TYPE_WITH_PRIVATE (ClutterTextBuffer, clutter_text_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* clutter_text_buffer_normal_get_text (ClutterTextBuffer *buffer, gsize *n_bytes) { ClutterTextBufferPrivate *priv = clutter_text_buffer_get_instance_private (buffer); if (n_bytes) *n_bytes = priv->normal_text_bytes; if (!priv->normal_text) return ""; return priv->normal_text; } static guint clutter_text_buffer_normal_get_length (ClutterTextBuffer *buffer) { ClutterTextBufferPrivate *priv = clutter_text_buffer_get_instance_private (buffer); return priv->normal_text_chars; } static guint clutter_text_buffer_normal_insert_text (ClutterTextBuffer *buffer, guint position, const gchar *chars, guint n_chars) { ClutterTextBufferPrivate *pv = clutter_text_buffer_get_instance_private (buffer);; 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 < CLUTTER_TEXT_BUFFER_MAX_SIZE) pv->normal_text_size *= 2; else { pv->normal_text_size = CLUTTER_TEXT_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'; clutter_text_buffer_emit_inserted_text (buffer, position, chars, n_chars); return n_chars; } static guint clutter_text_buffer_normal_delete_text (ClutterTextBuffer *buffer, guint position, guint n_chars) { ClutterTextBufferPrivate *pv = clutter_text_buffer_get_instance_private (buffer);; 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); clutter_text_buffer_emit_deleted_text (buffer, position, n_chars); } return n_chars; } /* -------------------------------------------------------------------------------- * */ static void clutter_text_buffer_real_inserted_text (ClutterTextBuffer *buffer, guint position, const gchar *chars, guint n_chars) { g_object_notify_by_pspec (G_OBJECT (buffer), obj_props[PROP_TEXT]); g_object_notify_by_pspec (G_OBJECT (buffer), obj_props[PROP_LENGTH]); } static void clutter_text_buffer_real_deleted_text (ClutterTextBuffer *buffer, guint position, guint n_chars) { g_object_notify_by_pspec (G_OBJECT (buffer), obj_props[PROP_TEXT]); g_object_notify_by_pspec (G_OBJECT (buffer), obj_props[PROP_LENGTH]); } /* -------------------------------------------------------------------------------- * */ static void clutter_text_buffer_init (ClutterTextBuffer *self) { ClutterTextBufferPrivate *priv = clutter_text_buffer_get_instance_private (self); priv->normal_text = NULL; priv->normal_text_chars = 0; priv->normal_text_bytes = 0; priv->normal_text_size = 0; } static void clutter_text_buffer_finalize (GObject *obj) { ClutterTextBuffer *buffer = CLUTTER_TEXT_BUFFER (obj); ClutterTextBufferPrivate *pv = clutter_text_buffer_get_instance_private (buffer); 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 (clutter_text_buffer_parent_class)->finalize (obj); } static void clutter_text_buffer_set_property (GObject *obj, guint prop_id, const GValue *value, GParamSpec *pspec) { ClutterTextBuffer *buffer = CLUTTER_TEXT_BUFFER (obj); switch (prop_id) { case PROP_MAX_LENGTH: clutter_text_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 clutter_text_buffer_get_property (GObject *obj, guint prop_id, GValue *value, GParamSpec *pspec) { ClutterTextBuffer *buffer = CLUTTER_TEXT_BUFFER (obj); switch (prop_id) { case PROP_TEXT: g_value_set_string (value, clutter_text_buffer_get_text (buffer)); break; case PROP_LENGTH: g_value_set_uint (value, clutter_text_buffer_get_length (buffer)); break; case PROP_MAX_LENGTH: g_value_set_int (value, clutter_text_buffer_get_max_length (buffer)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (obj, prop_id, pspec); break; } } static void clutter_text_buffer_class_init (ClutterTextBufferClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); gobject_class->finalize = clutter_text_buffer_finalize; gobject_class->set_property = clutter_text_buffer_set_property; gobject_class->get_property = clutter_text_buffer_get_property; klass->get_text = clutter_text_buffer_normal_get_text; klass->get_length = clutter_text_buffer_normal_get_length; klass->insert_text = clutter_text_buffer_normal_insert_text; klass->delete_text = clutter_text_buffer_normal_delete_text; klass->inserted_text = clutter_text_buffer_real_inserted_text; klass->deleted_text = clutter_text_buffer_real_deleted_text; /** * ClutterTextBuffer:text: * * The contents of the buffer. */ obj_props[PROP_TEXT] = g_param_spec_string ("text", NULL, NULL, "", G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); /** * ClutterTextBuffer:length: * * The length (in characters) of the text in buffer. */ obj_props[PROP_LENGTH] = g_param_spec_uint ("length", NULL, NULL, 0, CLUTTER_TEXT_BUFFER_MAX_SIZE, 0, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); /** * ClutterTextBuffer:max-length: * * The maximum length (in characters) of the text in the buffer. */ obj_props[PROP_MAX_LENGTH] = g_param_spec_int ("max-length", NULL, NULL, 0, CLUTTER_TEXT_BUFFER_MAX_SIZE, 0, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_properties (gobject_class, PROP_LAST, obj_props); /** * ClutterTextBuffer::inserted-text: * @buffer: a #ClutterTextBuffer * @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. */ signals[INSERTED_TEXT] = g_signal_new (I_("inserted-text"), CLUTTER_TYPE_TEXT_BUFFER, G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (ClutterTextBufferClass, inserted_text), NULL, NULL, _clutter_marshal_VOID__UINT_STRING_UINT, G_TYPE_NONE, 3, G_TYPE_UINT, G_TYPE_STRING, G_TYPE_UINT); /** * ClutterTextBuffer::deleted-text: * @buffer: a #ClutterTextBuffer * @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. */ signals[DELETED_TEXT] = g_signal_new (I_("deleted-text"), CLUTTER_TYPE_TEXT_BUFFER, G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (ClutterTextBufferClass, deleted_text), NULL, NULL, _clutter_marshal_VOID__UINT_UINT, G_TYPE_NONE, 2, G_TYPE_UINT, G_TYPE_UINT); } /* -------------------------------------------------------------------------------- * */ /** * clutter_text_buffer_new: * * Create a new ClutterTextBuffer object. * * Return value: A new ClutterTextBuffer object. **/ ClutterTextBuffer* clutter_text_buffer_new (void) { return g_object_new (CLUTTER_TYPE_TEXT_BUFFER, NULL); } /** * clutter_text_buffer_new_with_text: * @text: (allow-none): initial buffer text * @text_len: initial buffer text length, or -1 for null-terminated. * * Create a new ClutterTextBuffer object with some text. * * Return value: A new ClutterTextBuffer object. **/ ClutterTextBuffer* clutter_text_buffer_new_with_text (const gchar *text, gssize text_len) { ClutterTextBuffer *buffer; buffer = clutter_text_buffer_new (); clutter_text_buffer_set_text (buffer, text, text_len); return buffer; } /** * clutter_text_buffer_get_length: * @buffer: a #ClutterTextBuffer * * Retrieves the length in characters of the buffer. * * Return value: The number of characters in the buffer. **/ guint clutter_text_buffer_get_length (ClutterTextBuffer *buffer) { ClutterTextBufferClass *klass; g_return_val_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer), 0); klass = CLUTTER_TEXT_BUFFER_GET_CLASS (buffer); g_return_val_if_fail (klass->get_length != NULL, 0); return (*klass->get_length) (buffer); } /** * clutter_text_buffer_get_bytes: * @buffer: a #ClutterTextBuffer * * Retrieves the length in bytes of the buffer. * See [method@TextBuffer.get_length]. * * Return value: The byte length of the buffer. **/ gsize clutter_text_buffer_get_bytes (ClutterTextBuffer *buffer) { ClutterTextBufferClass *klass; gsize bytes = 0; g_return_val_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer), 0); klass = CLUTTER_TEXT_BUFFER_GET_CLASS (buffer); g_return_val_if_fail (klass->get_text != NULL, 0); (*klass->get_text) (buffer, &bytes); return bytes; } /** * clutter_text_buffer_get_text: * @buffer: a #ClutterTextBuffer * * 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. **/ const gchar* clutter_text_buffer_get_text (ClutterTextBuffer *buffer) { ClutterTextBufferClass *klass; g_return_val_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer), NULL); klass = CLUTTER_TEXT_BUFFER_GET_CLASS (buffer); g_return_val_if_fail (klass->get_text != NULL, NULL); return (*klass->get_text) (buffer, NULL); } /** * clutter_text_buffer_set_text: * @buffer: a #ClutterTextBuffer * @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 [method@TextBuffer.delete_text] * and [method@TextBuffer.insert_text]. * * Note that @n_chars is in characters, not in bytes. **/ void clutter_text_buffer_set_text (ClutterTextBuffer *buffer, const gchar *chars, gint n_chars) { g_return_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer)); g_return_if_fail (chars != NULL); g_object_freeze_notify (G_OBJECT (buffer)); clutter_text_buffer_delete_text (buffer, 0, -1); clutter_text_buffer_insert_text (buffer, 0, chars, n_chars); g_object_thaw_notify (G_OBJECT (buffer)); } /** * clutter_text_buffer_set_max_length: * @buffer: a #ClutterTextBuffer * @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, %CLUTTER_TEXT_BUFFER_MAX_SIZE ]. * * 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. **/ void clutter_text_buffer_set_max_length (ClutterTextBuffer *buffer, gint max_length) { ClutterTextBufferPrivate *priv; g_return_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer)); priv = clutter_text_buffer_get_instance_private (buffer); max_length = CLAMP (max_length, 0, CLUTTER_TEXT_BUFFER_MAX_SIZE); if (max_length > 0 && clutter_text_buffer_get_length (buffer) > max_length) clutter_text_buffer_delete_text (buffer, max_length, -1); priv->max_length = max_length; g_object_notify_by_pspec (G_OBJECT (buffer), obj_props[PROP_MAX_LENGTH]); } /** * clutter_text_buffer_get_max_length: * @buffer: a #ClutterTextBuffer * * Retrieves the maximum allowed length of the text in * @buffer. See [method@TextBuffer.set_max_length]. * * Return value: the maximum allowed number of characters * in #ClutterTextBuffer, or 0 if there is no maximum. */ gint clutter_text_buffer_get_max_length (ClutterTextBuffer *buffer) { ClutterTextBufferPrivate *priv; g_return_val_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer), 0); priv = clutter_text_buffer_get_instance_private (buffer); return priv->max_length; } /** * clutter_text_buffer_insert_text: * @buffer: a #ClutterTextBuffer * @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. */ guint clutter_text_buffer_insert_text (ClutterTextBuffer *buffer, guint position, const gchar *chars, gint n_chars) { ClutterTextBufferClass *klass; ClutterTextBufferPrivate *pv; guint length; g_return_val_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer), 0); length = clutter_text_buffer_get_length (buffer); pv = clutter_text_buffer_get_instance_private (buffer);; 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 = CLUTTER_TEXT_BUFFER_GET_CLASS (buffer); g_return_val_if_fail (klass->insert_text != NULL, 0); return (klass->insert_text) (buffer, position, chars, n_chars); } /** * clutter_text_buffer_delete_text: * @buffer: a #ClutterTextBuffer * @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. */ guint clutter_text_buffer_delete_text (ClutterTextBuffer *buffer, guint position, gint n_chars) { ClutterTextBufferClass *klass; guint length; g_return_val_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer), 0); length = clutter_text_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 = CLUTTER_TEXT_BUFFER_GET_CLASS (buffer); g_return_val_if_fail (klass->delete_text != NULL, 0); return (klass->delete_text) (buffer, position, n_chars); } /** * clutter_text_buffer_emit_inserted_text: * @buffer: a #ClutterTextBuffer * @position: position at which text was inserted * @chars: text that was inserted * @n_chars: number of characters inserted * * Emits the [signal@TextBuffer::inserted-text] signal on @buffer. * * Used when subclassing #ClutterTextBuffer */ void clutter_text_buffer_emit_inserted_text (ClutterTextBuffer *buffer, guint position, const gchar *chars, guint n_chars) { g_return_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer)); g_signal_emit (buffer, signals[INSERTED_TEXT], 0, position, chars, n_chars); } /** * clutter_text_buffer_emit_deleted_text: * @buffer: a #ClutterTextBuffer * @position: position at which text was deleted * @n_chars: number of characters deleted * * Emits the [signal@TextBuffer::deleted-text] signal on @buffer. * * Used when subclassing #ClutterTextBuffer */ void clutter_text_buffer_emit_deleted_text (ClutterTextBuffer *buffer, guint position, guint n_chars) { g_return_if_fail (CLUTTER_IS_TEXT_BUFFER (buffer)); g_signal_emit (buffer, signals[DELETED_TEXT], 0, position, n_chars); }