2003-09-06 Michael Natterer <mitch@gimp.org> To optimize duplicate and/or wrong image updates away, introduced new policy that a child object must never explicitly update or invalidate its parent object (just like the GUI is not updated explicitly by the core): * app/core/gimpdrawable.[ch]: added new signal GimpDrawable::update(). Never update or invalidate the image when the drawable is updated or invalidated. (gimp_drawable_set_visible): don't gimp_drawable_update() the drawable since its pixels have not changed. * app/core/gimpimage.[ch]: connect to the "add" and "remove" signals of the layers and channels containers. Also connect to the "update" and "visibility_changed" signals of all drawables in these containers (optimizes away updates issued by drawables which are not yet added to the image and updates of the selection mask). Also, don't propagate updates to the image if the emitting drawable is invisible (optimizes away updates issued by invisible drawables). (gimp_image_add_layer,channel) (gimp_image_remove_layer,channel): don't update the image since that's done by our "add" and "remove" handlers now. (gimp_image_position_layer,channel): update just the image, not the drawable since its pixels have not changed. (gimp_image_real_colormap_changed) (gimp_image_set_component_visible): always call gimp_image_update() *and* gimp_viewable_invalidate_preview() to get everything updated, since update and invalidate of images are not connected. * app/core/gimpimage-undo-push.c (undo_pop_layer,channel): don't update the drawable since (a) its pixels don't change and (b) the image updates itself upon adding/removing now. (undo_pop_layer_mod): replaced gimp_image_update() by gimp_drawable_update() (just for consistency with other similar functions). * app/core/gimplayer.c: connect to "update" of the layer mask and issue updates on the layer if the mask update has any effect on the projection. (gimp_layer_create_mask): don't set the mask's offsets here since they may be different when we later add the mask to the layer. * app/core/gimplayermask.c (gimp_layer_mask_set_layer): set the mask offsets here instead. * app/core/gimpchannel.c (gimp_channel_translate): update the channel even if push_undo == FALSE. * app/paint/gimppaintcore.c (gimp_paint_core_finish) * app/tools/gimpinktool.c (ink_finish): invalidate both the drawable and the image preview since invalidating the drawable doesn't invalidate the image any more. * app/text/gimptextlayer.c (gimp_text_layer_render_now): also update the new extents of the text layer, not only the old one. (gimp_text_layer_render_layout): don't update the drawable since gimp_drawable_fill() already updated it.
519 lines
15 KiB
C
519 lines
15 KiB
C
/* The GIMP -- an image manipulation program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* GimpTextLayer
|
|
* Copyright (C) 2002-2003 Sven Neumann <sven@gimp.org>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include <glib-object.h>
|
|
#include <pango/pangoft2.h>
|
|
|
|
#include "text-types.h"
|
|
|
|
#include "base/pixel-region.h"
|
|
#include "base/tile-manager.h"
|
|
|
|
#include "paint-funcs/paint-funcs.h"
|
|
|
|
#include "config/gimpconfig.h"
|
|
|
|
#include "core/gimp.h"
|
|
#include "core/gimpimage.h"
|
|
|
|
#include "gimptext.h"
|
|
#include "gimptext-bitmap.h"
|
|
#include "gimptext-private.h"
|
|
#include "gimptextlayer.h"
|
|
#include "gimptextlayer-transform.h"
|
|
#include "gimptextlayout.h"
|
|
#include "gimptextlayout-render.h"
|
|
|
|
#include "gimp-intl.h"
|
|
|
|
|
|
static void gimp_text_layer_class_init (GimpTextLayerClass *klass);
|
|
static void gimp_text_layer_init (GimpTextLayer *layer);
|
|
static void gimp_text_layer_dispose (GObject *object);
|
|
|
|
static gsize gimp_text_layer_get_memsize (GimpObject *object,
|
|
gsize *gui_size);
|
|
static TempBuf * gimp_text_layer_get_preview (GimpViewable *viewable,
|
|
gint width,
|
|
gint height);
|
|
static GimpItem * gimp_text_layer_duplicate (GimpItem *item,
|
|
GType new_type,
|
|
gboolean add_alpha);
|
|
static void gimp_text_layer_rename (GimpItem *item,
|
|
const gchar *new_name,
|
|
const gchar *undo_desc);
|
|
|
|
static void gimp_text_layer_set_text (GimpTextLayer *layer,
|
|
GimpText *text);
|
|
static void gimp_text_layer_notify_text (GimpTextLayer *layer);
|
|
static gboolean gimp_text_layer_idle_render (GimpTextLayer *layer);
|
|
static gboolean gimp_text_layer_render_now (GimpTextLayer *layer);
|
|
static void gimp_text_layer_render_layout (GimpTextLayer *layer,
|
|
GimpTextLayout *layout);
|
|
|
|
|
|
static GimpLayerClass *parent_class = NULL;
|
|
|
|
|
|
GType
|
|
gimp_text_layer_get_type (void)
|
|
{
|
|
static GType layer_type = 0;
|
|
|
|
if (! layer_type)
|
|
{
|
|
static const GTypeInfo layer_info =
|
|
{
|
|
sizeof (GimpTextLayerClass),
|
|
(GBaseInitFunc) NULL,
|
|
(GBaseFinalizeFunc) NULL,
|
|
(GClassInitFunc) gimp_text_layer_class_init,
|
|
NULL, /* class_finalize */
|
|
NULL, /* class_data */
|
|
sizeof (GimpTextLayer),
|
|
0, /* n_preallocs */
|
|
(GInstanceInitFunc) gimp_text_layer_init,
|
|
};
|
|
|
|
layer_type = g_type_register_static (GIMP_TYPE_LAYER,
|
|
"GimpTextLayer",
|
|
&layer_info, 0);
|
|
}
|
|
|
|
return layer_type;
|
|
}
|
|
|
|
static void
|
|
gimp_text_layer_class_init (GimpTextLayerClass *klass)
|
|
{
|
|
GObjectClass *object_class;
|
|
GimpObjectClass *gimp_object_class;
|
|
GimpViewableClass *viewable_class;
|
|
GimpItemClass *item_class;
|
|
|
|
object_class = G_OBJECT_CLASS (klass);
|
|
gimp_object_class = GIMP_OBJECT_CLASS (klass);
|
|
viewable_class = GIMP_VIEWABLE_CLASS (klass);
|
|
item_class = GIMP_ITEM_CLASS (klass);
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
object_class->dispose = gimp_text_layer_dispose;
|
|
|
|
gimp_object_class->get_memsize = gimp_text_layer_get_memsize;
|
|
|
|
viewable_class->default_stock_id = "gimp-text-layer";
|
|
viewable_class->get_preview = gimp_text_layer_get_preview;
|
|
|
|
item_class->default_name = _("Text Layer");
|
|
item_class->duplicate = gimp_text_layer_duplicate;
|
|
item_class->rename = gimp_text_layer_rename;
|
|
|
|
#if 0
|
|
item_class->scale = gimp_text_layer_scale;
|
|
item_class->flip = gimp_text_layer_flip;
|
|
item_class->rotate = gimp_text_layer_rotate;
|
|
item_class->transform = gimp_text_layer_transform;
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
gimp_text_layer_init (GimpTextLayer *layer)
|
|
{
|
|
layer->text = NULL;
|
|
layer->idle_render_id = 0;
|
|
layer->auto_rename = TRUE;
|
|
}
|
|
|
|
static void
|
|
gimp_text_layer_dispose (GObject *object)
|
|
{
|
|
GimpTextLayer *layer;
|
|
|
|
layer = GIMP_TEXT_LAYER (object);
|
|
|
|
if (layer->idle_render_id)
|
|
{
|
|
g_source_remove (layer->idle_render_id);
|
|
layer->idle_render_id = 0;
|
|
}
|
|
if (layer->text)
|
|
{
|
|
g_object_unref (layer->text);
|
|
layer->text = NULL;
|
|
}
|
|
|
|
G_OBJECT_CLASS (parent_class)->dispose (object);
|
|
}
|
|
|
|
static gsize
|
|
gimp_text_layer_get_memsize (GimpObject *object,
|
|
gsize *gui_size)
|
|
{
|
|
GimpTextLayer *text_layer;
|
|
gsize memsize = 0;
|
|
|
|
text_layer = GIMP_TEXT_LAYER (object);
|
|
|
|
if (text_layer->text)
|
|
memsize += gimp_object_get_memsize (GIMP_OBJECT (text_layer->text),
|
|
gui_size);
|
|
|
|
return memsize + GIMP_OBJECT_CLASS (parent_class)->get_memsize (object,
|
|
gui_size);
|
|
}
|
|
|
|
static TempBuf *
|
|
gimp_text_layer_get_preview (GimpViewable *viewable,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
return GIMP_VIEWABLE_CLASS (parent_class)->get_preview (viewable,
|
|
width, height);
|
|
}
|
|
|
|
static GimpItem *
|
|
gimp_text_layer_duplicate (GimpItem *item,
|
|
GType new_type,
|
|
gboolean add_alpha)
|
|
{
|
|
GimpTextLayer *text_layer;
|
|
GimpItem *new_item;
|
|
GimpTextLayer *new_text_layer;
|
|
|
|
g_return_val_if_fail (g_type_is_a (new_type, GIMP_TYPE_DRAWABLE), NULL);
|
|
|
|
new_item = GIMP_ITEM_CLASS (parent_class)->duplicate (item, new_type,
|
|
add_alpha);
|
|
|
|
if (! GIMP_IS_TEXT_LAYER (new_item))
|
|
return new_item;
|
|
|
|
text_layer = GIMP_TEXT_LAYER (item);
|
|
new_text_layer = GIMP_TEXT_LAYER (new_item);
|
|
|
|
new_text_layer->text =
|
|
GIMP_TEXT (gimp_config_duplicate (G_OBJECT (text_layer->text)));
|
|
|
|
g_signal_connect_object (new_text_layer->text, "notify",
|
|
G_CALLBACK (gimp_text_layer_notify_text),
|
|
new_text_layer, G_CONNECT_SWAPPED);
|
|
|
|
return new_item;
|
|
}
|
|
|
|
static void
|
|
gimp_text_layer_rename (GimpItem *item,
|
|
const gchar *new_name,
|
|
const gchar *undo_desc)
|
|
{
|
|
GIMP_TEXT_LAYER (item)->auto_rename = FALSE;
|
|
|
|
GIMP_ITEM_CLASS (parent_class)->rename (item, new_name, undo_desc);
|
|
}
|
|
|
|
/**
|
|
* gimp_text_layer_new:
|
|
* @image: the #GimpImage the layer should belong to
|
|
* @text: a #GimpText object
|
|
*
|
|
* Creates a new text layer.
|
|
*
|
|
* Return value: a new #GimpTextLayer or %NULL in case of a problem
|
|
**/
|
|
GimpLayer *
|
|
gimp_text_layer_new (GimpImage *image,
|
|
GimpText *text)
|
|
{
|
|
GimpTextLayer *layer;
|
|
|
|
g_return_val_if_fail (GIMP_IS_IMAGE (image), NULL);
|
|
g_return_val_if_fail (GIMP_IS_TEXT (text), NULL);
|
|
|
|
if (!text->text)
|
|
return NULL;
|
|
|
|
layer = g_object_new (GIMP_TYPE_TEXT_LAYER, NULL);
|
|
|
|
gimp_drawable_configure (GIMP_DRAWABLE (layer),
|
|
image,
|
|
0, 0, 0, 0,
|
|
gimp_image_base_type_with_alpha (image),
|
|
NULL);
|
|
|
|
gimp_text_layer_set_text (layer, text);
|
|
|
|
if (! gimp_text_layer_render_now (layer))
|
|
{
|
|
g_object_unref (layer);
|
|
return NULL;
|
|
}
|
|
|
|
return GIMP_LAYER (layer);
|
|
}
|
|
|
|
/**
|
|
* gimp_text_layer_from_layer:
|
|
* @layer: a #GimpLayer object
|
|
* @text: a #GimpText object
|
|
*
|
|
* Converts a standard #GimpLayer and a #GimpText object into a
|
|
* #GimpTextLayer. The new text layer takes ownership of the @text and
|
|
* @layer objects. The @layer object is rendered unusable by this
|
|
* function. Don't even try to use if afterwards!
|
|
*
|
|
* This is a gross hack that is needed in order to load text layers
|
|
* from XCF files in a backwards-compatible way. Please don't use it
|
|
* for anything else!
|
|
*
|
|
* Return value: a newly allocated #GimpTextLayer object
|
|
**/
|
|
GimpLayer *
|
|
gimp_text_layer_from_layer (GimpLayer *layer,
|
|
GimpText *text)
|
|
{
|
|
GimpTextLayer *text_layer;
|
|
GimpItem *item;
|
|
GimpDrawable *drawable;
|
|
|
|
g_return_val_if_fail (GIMP_IS_LAYER (layer), NULL);
|
|
g_return_val_if_fail (GIMP_IS_TEXT (text), NULL);
|
|
|
|
text_layer = g_object_new (GIMP_TYPE_TEXT_LAYER, NULL);
|
|
|
|
item = GIMP_ITEM (text_layer);
|
|
drawable = GIMP_DRAWABLE (text_layer);
|
|
|
|
gimp_object_set_name (GIMP_OBJECT (text_layer),
|
|
gimp_object_get_name (GIMP_OBJECT (layer)));
|
|
|
|
item->ID = gimp_item_get_ID (GIMP_ITEM (layer));
|
|
item->tattoo = gimp_item_get_tattoo (GIMP_ITEM (layer));
|
|
item->gimage = gimp_item_get_image (GIMP_ITEM (layer));
|
|
|
|
gimp_item_set_image (GIMP_ITEM (layer), NULL);
|
|
g_hash_table_replace (item->gimage->gimp->item_table,
|
|
GINT_TO_POINTER (item->ID),
|
|
item);
|
|
|
|
item->parasites = GIMP_ITEM (layer)->parasites;
|
|
GIMP_ITEM (layer)->parasites = NULL;
|
|
|
|
item->width = gimp_item_width (GIMP_ITEM (layer));
|
|
item->height = gimp_item_height (GIMP_ITEM (layer));
|
|
|
|
gimp_item_offsets (GIMP_ITEM (layer), &item->offset_x, &item->offset_y);
|
|
|
|
item->linked = gimp_item_get_linked (GIMP_ITEM (layer));
|
|
|
|
drawable->tiles = GIMP_DRAWABLE (layer)->tiles;
|
|
GIMP_DRAWABLE (layer)->tiles = NULL;
|
|
|
|
drawable->visible = gimp_drawable_get_visible (GIMP_DRAWABLE (layer));
|
|
drawable->bytes = gimp_drawable_bytes (GIMP_DRAWABLE (layer));
|
|
drawable->type = gimp_drawable_type (GIMP_DRAWABLE (layer));
|
|
drawable->has_alpha = gimp_drawable_has_alpha (GIMP_DRAWABLE (layer));
|
|
|
|
GIMP_LAYER (text_layer)->opacity = gimp_layer_get_opacity (layer);
|
|
GIMP_LAYER (text_layer)->mode = gimp_layer_get_mode (layer);
|
|
GIMP_LAYER (text_layer)->preserve_trans = gimp_layer_get_preserve_trans (layer);
|
|
|
|
gimp_text_layer_set_text (text_layer, text);
|
|
|
|
g_object_unref (layer);
|
|
g_object_unref (text);
|
|
|
|
return GIMP_LAYER (text_layer);
|
|
}
|
|
|
|
static void
|
|
gimp_text_layer_set_text (GimpTextLayer *layer,
|
|
GimpText *text)
|
|
{
|
|
g_return_if_fail (GIMP_IS_TEXT_LAYER (layer));
|
|
g_return_if_fail (GIMP_IS_TEXT (text));
|
|
g_return_if_fail (layer->text == NULL);
|
|
|
|
layer->text = g_object_ref (text);
|
|
g_signal_connect_object (text, "notify",
|
|
G_CALLBACK (gimp_text_layer_notify_text),
|
|
layer, G_CONNECT_SWAPPED);
|
|
}
|
|
|
|
GimpText *
|
|
gimp_text_layer_get_text (GimpTextLayer *layer)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_TEXT_LAYER (layer), NULL);
|
|
|
|
return layer->text;
|
|
}
|
|
|
|
void
|
|
gimp_text_layer_render (GimpTextLayer *layer)
|
|
{
|
|
g_return_if_fail (GIMP_IS_TEXT_LAYER (layer));
|
|
|
|
if (layer->idle_render_id)
|
|
{
|
|
g_source_remove (layer->idle_render_id);
|
|
layer->idle_render_id = 0;
|
|
}
|
|
|
|
gimp_text_layer_render_now (layer);
|
|
}
|
|
|
|
static void
|
|
gimp_text_layer_notify_text (GimpTextLayer *layer)
|
|
{
|
|
if (layer->idle_render_id)
|
|
g_source_remove (layer->idle_render_id);
|
|
|
|
layer->idle_render_id =
|
|
g_idle_add_full (G_PRIORITY_LOW,
|
|
(GSourceFunc) gimp_text_layer_idle_render, layer,
|
|
NULL);
|
|
}
|
|
|
|
static gboolean
|
|
gimp_text_layer_idle_render (GimpTextLayer *layer)
|
|
{
|
|
layer->idle_render_id = 0;
|
|
|
|
gimp_text_layer_render_now (layer);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gimp_text_layer_render_now (GimpTextLayer *layer)
|
|
{
|
|
GimpDrawable *drawable;
|
|
GimpItem *item;
|
|
GimpImage *image;
|
|
GimpTextLayout *layout;
|
|
gint width;
|
|
gint height;
|
|
|
|
drawable = GIMP_DRAWABLE (layer);
|
|
item = GIMP_ITEM (layer);
|
|
image = gimp_item_get_image (item);
|
|
|
|
layout = gimp_text_layout_new (layer->text, image);
|
|
|
|
if (gimp_text_layout_get_size (layout, &width, &height))
|
|
{
|
|
if (width != gimp_item_width (item) ||
|
|
height != gimp_item_height (item))
|
|
{
|
|
gimp_drawable_update (drawable,
|
|
0, 0,
|
|
gimp_item_width (item),
|
|
gimp_item_height (item));
|
|
|
|
/* Make sure we're not caching any old selection info */
|
|
gimp_drawable_invalidate_boundary (GIMP_DRAWABLE (layer));
|
|
|
|
GIMP_ITEM (drawable)->width = width;
|
|
GIMP_ITEM (drawable)->height = height;
|
|
|
|
if (drawable->tiles)
|
|
tile_manager_unref (drawable->tiles);
|
|
|
|
drawable->tiles = tile_manager_new (width, height, drawable->bytes);
|
|
|
|
gimp_drawable_update (drawable,
|
|
0, 0,
|
|
gimp_item_width (item),
|
|
gimp_item_height (item));
|
|
|
|
gimp_viewable_size_changed (GIMP_VIEWABLE (layer));
|
|
}
|
|
}
|
|
|
|
if (layer->auto_rename)
|
|
gimp_object_set_name_safe (GIMP_OBJECT (layer),
|
|
layer->text->text ?
|
|
layer->text->text : _("Empty Text Layer"));
|
|
|
|
gimp_text_layer_render_layout (layer, layout);
|
|
g_object_unref (layout);
|
|
|
|
gimp_image_flush (image);
|
|
|
|
return (width > 0 && height > 0);
|
|
}
|
|
|
|
static void
|
|
gimp_text_layer_render_layout (GimpTextLayer *layer,
|
|
GimpTextLayout *layout)
|
|
{
|
|
GimpDrawable *drawable;
|
|
GimpItem *item;
|
|
TileManager *mask;
|
|
FT_Bitmap bitmap;
|
|
PixelRegion textPR;
|
|
PixelRegion maskPR;
|
|
gint i;
|
|
|
|
drawable = GIMP_DRAWABLE (layer);
|
|
item = GIMP_ITEM (layer);
|
|
|
|
gimp_drawable_fill (drawable, &layer->text->color);
|
|
|
|
bitmap.width = gimp_item_width (item);
|
|
bitmap.rows = gimp_item_height (item);
|
|
bitmap.pitch = bitmap.width;
|
|
if (bitmap.pitch & 3)
|
|
bitmap.pitch += 4 - (bitmap.pitch & 3);
|
|
|
|
bitmap.buffer = g_malloc0 (bitmap.rows * bitmap.pitch);
|
|
|
|
gimp_text_layout_render (layout,
|
|
(GimpTextRenderFunc) gimp_text_render_bitmap,
|
|
&bitmap);
|
|
|
|
mask = tile_manager_new (bitmap.width, bitmap.rows, 1);
|
|
pixel_region_init (&maskPR, mask, 0, 0, bitmap.width, bitmap.rows, TRUE);
|
|
|
|
for (i = 0; i < bitmap.rows; i++)
|
|
pixel_region_set_row (&maskPR,
|
|
0, i, bitmap.width,
|
|
bitmap.buffer + i * bitmap.pitch);
|
|
|
|
g_free (bitmap.buffer);
|
|
|
|
pixel_region_init (&textPR, drawable->tiles,
|
|
0, 0, bitmap.width, bitmap.rows, TRUE);
|
|
pixel_region_init (&maskPR, mask,
|
|
0, 0, bitmap.width, bitmap.rows, FALSE);
|
|
|
|
apply_mask_to_region (&textPR, &maskPR, OPAQUE_OPACITY);
|
|
|
|
tile_manager_unref (mask);
|
|
|
|
/* no need to gimp_drawable_update() since gimp_drawable_fill()
|
|
* did that for us.
|
|
*/
|
|
}
|