2001-05-07 15:58:47 +00:00
|
|
|
|
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- */
|
|
|
|
|
/* GdkPixbuf library - animated gif support
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 1999 The Free Software Foundation
|
|
|
|
|
*
|
|
|
|
|
* Authors: Jonathan Blandford <jrb@redhat.com>
|
|
|
|
|
* Havoc Pennington <hp@redhat.com>
|
|
|
|
|
*
|
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
|
* modify it under the terms of the GNU Lesser 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
|
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
* License along with this library; if not, write to the
|
|
|
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
|
*/
|
|
|
|
|
|
2008-06-22 14:28:52 +00:00
|
|
|
|
#include "config.h"
|
2001-05-07 15:58:47 +00:00
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include "gdk-pixbuf-private.h"
|
|
|
|
|
#include "io-gif-animation.h"
|
|
|
|
|
|
|
|
|
|
static void gdk_pixbuf_gif_anim_class_init (GdkPixbufGifAnimClass *klass);
|
|
|
|
|
static void gdk_pixbuf_gif_anim_finalize (GObject *object);
|
|
|
|
|
|
|
|
|
|
static gboolean gdk_pixbuf_gif_anim_is_static_image (GdkPixbufAnimation *animation);
|
|
|
|
|
static GdkPixbuf* gdk_pixbuf_gif_anim_get_static_image (GdkPixbufAnimation *animation);
|
|
|
|
|
|
|
|
|
|
static void gdk_pixbuf_gif_anim_get_size (GdkPixbufAnimation *anim,
|
|
|
|
|
int *width,
|
|
|
|
|
int *height);
|
|
|
|
|
static GdkPixbufAnimationIter* gdk_pixbuf_gif_anim_get_iter (GdkPixbufAnimation *anim,
|
|
|
|
|
const GTimeVal *start_time);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpointer parent_class;
|
|
|
|
|
|
|
|
|
|
GType
|
|
|
|
|
gdk_pixbuf_gif_anim_get_type (void)
|
|
|
|
|
{
|
|
|
|
|
static GType object_type = 0;
|
|
|
|
|
|
|
|
|
|
if (!object_type) {
|
|
|
|
|
static const GTypeInfo object_info = {
|
|
|
|
|
sizeof (GdkPixbufGifAnimClass),
|
|
|
|
|
(GBaseInitFunc) NULL,
|
|
|
|
|
(GBaseFinalizeFunc) NULL,
|
|
|
|
|
(GClassInitFunc) gdk_pixbuf_gif_anim_class_init,
|
|
|
|
|
NULL, /* class_finalize */
|
|
|
|
|
NULL, /* class_data */
|
|
|
|
|
sizeof (GdkPixbufGifAnim),
|
|
|
|
|
0, /* n_preallocs */
|
|
|
|
|
(GInstanceInitFunc) NULL,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
object_type = g_type_register_static (GDK_TYPE_PIXBUF_ANIMATION,
|
2005-08-31 15:18:41 +00:00
|
|
|
|
g_intern_static_string ("GdkPixbufGifAnim"),
|
2001-05-07 15:58:47 +00:00
|
|
|
|
&object_info, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return object_type;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gdk_pixbuf_gif_anim_class_init (GdkPixbufGifAnimClass *klass)
|
|
|
|
|
{
|
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
GdkPixbufAnimationClass *anim_class = GDK_PIXBUF_ANIMATION_CLASS (klass);
|
|
|
|
|
|
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
|
|
|
|
|
|
object_class->finalize = gdk_pixbuf_gif_anim_finalize;
|
|
|
|
|
|
|
|
|
|
anim_class->is_static_image = gdk_pixbuf_gif_anim_is_static_image;
|
|
|
|
|
anim_class->get_static_image = gdk_pixbuf_gif_anim_get_static_image;
|
|
|
|
|
anim_class->get_size = gdk_pixbuf_gif_anim_get_size;
|
|
|
|
|
anim_class->get_iter = gdk_pixbuf_gif_anim_get_iter;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gdk_pixbuf_gif_anim_finalize (GObject *object)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnim *gif_anim = GDK_PIXBUF_GIF_ANIM (object);
|
|
|
|
|
|
|
|
|
|
GList *l;
|
|
|
|
|
GdkPixbufFrame *frame;
|
|
|
|
|
|
|
|
|
|
for (l = gif_anim->frames; l; l = l->next) {
|
|
|
|
|
frame = l->data;
|
Remove assorted G_OBJECT casts where unnecessary.
2001-12-13 Sebastian Wilhelmi <wilhelmi@ira.uka.de>
* gdk-pixbuf-animation.c, gdk-pixbuf-loader.c, gdk-pixpuf.c,
io-gif-animation.c, io-gif.c, io-tiff.c, test-loaders.c: Remove
assorted G_OBJECT casts where unnecessary.
* gdk-pixbuf-loader.c: Call g_object_ref and g_object_unref
instead of gdk_pixbuf_animation_ref and gdk_pixbuf_animation_unref
resp.
* gdk-pixbuf-csource.c, io-bmp.c, io-gif-animation.c, io-ico.c,
io-jpeg.c, io-png.c, io-pnm.c, io-ras.c, io-tga.c, io-wbmp.c,
io-xbm.c, io-xpm.c, test-gdk-pixbuf.c: Dito for gdk_pixbuf_ref and
gdk_pixbuf_unref.
* Makefile.am, pixops/Makefile.am: Compile everything with
-DG_DISABLE_DEPRECATED -DGDK_PIXBUF_DISABLE_DEPRECATED
* gdk-pixdata.c: Use g_ascii_strup() instead of g_strup().
* io-xpm.c: Use g_ascii_strcasecmp() instead of g_strcasecmp().
* demos/testpixbuf-drawable.c, demos/testpixbuf-save.c,
demos/testpixbuf-scale.c, demos/testpixbuf.c: Call g_object_ref
and g_object_unref instead of gdk_pixbuf_ref and gdk_pixbuf_unref
resp.
2001-12-13 21:22:12 +00:00
|
|
|
|
g_object_unref (frame->pixbuf);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
if (frame->composited)
|
Remove assorted G_OBJECT casts where unnecessary.
2001-12-13 Sebastian Wilhelmi <wilhelmi@ira.uka.de>
* gdk-pixbuf-animation.c, gdk-pixbuf-loader.c, gdk-pixpuf.c,
io-gif-animation.c, io-gif.c, io-tiff.c, test-loaders.c: Remove
assorted G_OBJECT casts where unnecessary.
* gdk-pixbuf-loader.c: Call g_object_ref and g_object_unref
instead of gdk_pixbuf_animation_ref and gdk_pixbuf_animation_unref
resp.
* gdk-pixbuf-csource.c, io-bmp.c, io-gif-animation.c, io-ico.c,
io-jpeg.c, io-png.c, io-pnm.c, io-ras.c, io-tga.c, io-wbmp.c,
io-xbm.c, io-xpm.c, test-gdk-pixbuf.c: Dito for gdk_pixbuf_ref and
gdk_pixbuf_unref.
* Makefile.am, pixops/Makefile.am: Compile everything with
-DG_DISABLE_DEPRECATED -DGDK_PIXBUF_DISABLE_DEPRECATED
* gdk-pixdata.c: Use g_ascii_strup() instead of g_strup().
* io-xpm.c: Use g_ascii_strcasecmp() instead of g_strcasecmp().
* demos/testpixbuf-drawable.c, demos/testpixbuf-save.c,
demos/testpixbuf-scale.c, demos/testpixbuf.c: Call g_object_ref
and g_object_unref instead of gdk_pixbuf_ref and gdk_pixbuf_unref
resp.
2001-12-13 21:22:12 +00:00
|
|
|
|
g_object_unref (frame->composited);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
if (frame->revert)
|
Remove assorted G_OBJECT casts where unnecessary.
2001-12-13 Sebastian Wilhelmi <wilhelmi@ira.uka.de>
* gdk-pixbuf-animation.c, gdk-pixbuf-loader.c, gdk-pixpuf.c,
io-gif-animation.c, io-gif.c, io-tiff.c, test-loaders.c: Remove
assorted G_OBJECT casts where unnecessary.
* gdk-pixbuf-loader.c: Call g_object_ref and g_object_unref
instead of gdk_pixbuf_animation_ref and gdk_pixbuf_animation_unref
resp.
* gdk-pixbuf-csource.c, io-bmp.c, io-gif-animation.c, io-ico.c,
io-jpeg.c, io-png.c, io-pnm.c, io-ras.c, io-tga.c, io-wbmp.c,
io-xbm.c, io-xpm.c, test-gdk-pixbuf.c: Dito for gdk_pixbuf_ref and
gdk_pixbuf_unref.
* Makefile.am, pixops/Makefile.am: Compile everything with
-DG_DISABLE_DEPRECATED -DGDK_PIXBUF_DISABLE_DEPRECATED
* gdk-pixdata.c: Use g_ascii_strup() instead of g_strup().
* io-xpm.c: Use g_ascii_strcasecmp() instead of g_strcasecmp().
* demos/testpixbuf-drawable.c, demos/testpixbuf-save.c,
demos/testpixbuf-scale.c, demos/testpixbuf.c: Call g_object_ref
and g_object_unref instead of gdk_pixbuf_ref and gdk_pixbuf_unref
resp.
2001-12-13 21:22:12 +00:00
|
|
|
|
g_object_unref (frame->revert);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
g_free (frame);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_list_free (gif_anim->frames);
|
|
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
gdk_pixbuf_gif_anim_is_static_image (GdkPixbufAnimation *animation)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnim *gif_anim;
|
|
|
|
|
|
|
|
|
|
gif_anim = GDK_PIXBUF_GIF_ANIM (animation);
|
|
|
|
|
|
|
|
|
|
return (gif_anim->frames != NULL &&
|
|
|
|
|
gif_anim->frames->next == NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static GdkPixbuf*
|
|
|
|
|
gdk_pixbuf_gif_anim_get_static_image (GdkPixbufAnimation *animation)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnim *gif_anim;
|
|
|
|
|
|
|
|
|
|
gif_anim = GDK_PIXBUF_GIF_ANIM (animation);
|
|
|
|
|
|
|
|
|
|
if (gif_anim->frames == NULL)
|
|
|
|
|
return NULL;
|
|
|
|
|
else
|
|
|
|
|
return GDK_PIXBUF (((GdkPixbufFrame*)gif_anim->frames->data)->pixbuf);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gdk_pixbuf_gif_anim_get_size (GdkPixbufAnimation *anim,
|
|
|
|
|
int *width,
|
|
|
|
|
int *height)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnim *gif_anim;
|
|
|
|
|
|
|
|
|
|
gif_anim = GDK_PIXBUF_GIF_ANIM (anim);
|
|
|
|
|
|
|
|
|
|
if (width)
|
|
|
|
|
*width = gif_anim->width;
|
|
|
|
|
|
|
|
|
|
if (height)
|
|
|
|
|
*height = gif_anim->height;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
iter_clear (GdkPixbufGifAnimIter *iter)
|
|
|
|
|
{
|
|
|
|
|
iter->current_frame = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
iter_restart (GdkPixbufGifAnimIter *iter)
|
|
|
|
|
{
|
|
|
|
|
iter_clear (iter);
|
|
|
|
|
|
|
|
|
|
iter->current_frame = iter->gif_anim->frames;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static GdkPixbufAnimationIter*
|
|
|
|
|
gdk_pixbuf_gif_anim_get_iter (GdkPixbufAnimation *anim,
|
|
|
|
|
const GTimeVal *start_time)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnimIter *iter;
|
|
|
|
|
|
|
|
|
|
iter = g_object_new (GDK_TYPE_PIXBUF_GIF_ANIM_ITER, NULL);
|
|
|
|
|
|
|
|
|
|
iter->gif_anim = GDK_PIXBUF_GIF_ANIM (anim);
|
|
|
|
|
|
Remove assorted G_OBJECT casts where unnecessary.
2001-12-13 Sebastian Wilhelmi <wilhelmi@ira.uka.de>
* gdk-pixbuf-animation.c, gdk-pixbuf-loader.c, gdk-pixpuf.c,
io-gif-animation.c, io-gif.c, io-tiff.c, test-loaders.c: Remove
assorted G_OBJECT casts where unnecessary.
* gdk-pixbuf-loader.c: Call g_object_ref and g_object_unref
instead of gdk_pixbuf_animation_ref and gdk_pixbuf_animation_unref
resp.
* gdk-pixbuf-csource.c, io-bmp.c, io-gif-animation.c, io-ico.c,
io-jpeg.c, io-png.c, io-pnm.c, io-ras.c, io-tga.c, io-wbmp.c,
io-xbm.c, io-xpm.c, test-gdk-pixbuf.c: Dito for gdk_pixbuf_ref and
gdk_pixbuf_unref.
* Makefile.am, pixops/Makefile.am: Compile everything with
-DG_DISABLE_DEPRECATED -DGDK_PIXBUF_DISABLE_DEPRECATED
* gdk-pixdata.c: Use g_ascii_strup() instead of g_strup().
* io-xpm.c: Use g_ascii_strcasecmp() instead of g_strcasecmp().
* demos/testpixbuf-drawable.c, demos/testpixbuf-save.c,
demos/testpixbuf-scale.c, demos/testpixbuf.c: Call g_object_ref
and g_object_unref instead of gdk_pixbuf_ref and gdk_pixbuf_unref
resp.
2001-12-13 21:22:12 +00:00
|
|
|
|
g_object_ref (iter->gif_anim);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
iter_restart (iter);
|
|
|
|
|
|
|
|
|
|
iter->start_time = *start_time;
|
|
|
|
|
iter->current_time = *start_time;
|
2003-05-19 21:44:03 +00:00
|
|
|
|
iter->first_loop_slowness = 0;
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
return GDK_PIXBUF_ANIMATION_ITER (iter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void gdk_pixbuf_gif_anim_iter_class_init (GdkPixbufGifAnimIterClass *klass);
|
|
|
|
|
static void gdk_pixbuf_gif_anim_iter_finalize (GObject *object);
|
|
|
|
|
|
|
|
|
|
static int gdk_pixbuf_gif_anim_iter_get_delay_time (GdkPixbufAnimationIter *iter);
|
|
|
|
|
static GdkPixbuf* gdk_pixbuf_gif_anim_iter_get_pixbuf (GdkPixbufAnimationIter *iter);
|
|
|
|
|
static gboolean gdk_pixbuf_gif_anim_iter_on_currently_loading_frame (GdkPixbufAnimationIter *iter);
|
|
|
|
|
static gboolean gdk_pixbuf_gif_anim_iter_advance (GdkPixbufAnimationIter *iter,
|
|
|
|
|
const GTimeVal *current_time);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpointer iter_parent_class;
|
|
|
|
|
|
|
|
|
|
GType
|
|
|
|
|
gdk_pixbuf_gif_anim_iter_get_type (void)
|
|
|
|
|
{
|
|
|
|
|
static GType object_type = 0;
|
|
|
|
|
|
|
|
|
|
if (!object_type) {
|
|
|
|
|
static const GTypeInfo object_info = {
|
|
|
|
|
sizeof (GdkPixbufGifAnimIterClass),
|
|
|
|
|
(GBaseInitFunc) NULL,
|
|
|
|
|
(GBaseFinalizeFunc) NULL,
|
|
|
|
|
(GClassInitFunc) gdk_pixbuf_gif_anim_iter_class_init,
|
|
|
|
|
NULL, /* class_finalize */
|
|
|
|
|
NULL, /* class_data */
|
|
|
|
|
sizeof (GdkPixbufGifAnimIter),
|
|
|
|
|
0, /* n_preallocs */
|
|
|
|
|
(GInstanceInitFunc) NULL,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
object_type = g_type_register_static (GDK_TYPE_PIXBUF_ANIMATION_ITER,
|
2005-08-31 15:18:41 +00:00
|
|
|
|
g_intern_static_string ("GdkPixbufGifAnimIter"),
|
2001-05-07 15:58:47 +00:00
|
|
|
|
&object_info, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return object_type;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gdk_pixbuf_gif_anim_iter_class_init (GdkPixbufGifAnimIterClass *klass)
|
|
|
|
|
{
|
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
GdkPixbufAnimationIterClass *anim_iter_class =
|
|
|
|
|
GDK_PIXBUF_ANIMATION_ITER_CLASS (klass);
|
|
|
|
|
|
|
|
|
|
iter_parent_class = g_type_class_peek_parent (klass);
|
|
|
|
|
|
|
|
|
|
object_class->finalize = gdk_pixbuf_gif_anim_iter_finalize;
|
|
|
|
|
|
|
|
|
|
anim_iter_class->get_delay_time = gdk_pixbuf_gif_anim_iter_get_delay_time;
|
|
|
|
|
anim_iter_class->get_pixbuf = gdk_pixbuf_gif_anim_iter_get_pixbuf;
|
|
|
|
|
anim_iter_class->on_currently_loading_frame = gdk_pixbuf_gif_anim_iter_on_currently_loading_frame;
|
|
|
|
|
anim_iter_class->advance = gdk_pixbuf_gif_anim_iter_advance;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
gdk_pixbuf_gif_anim_iter_finalize (GObject *object)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnimIter *iter = GDK_PIXBUF_GIF_ANIM_ITER (object);
|
|
|
|
|
|
|
|
|
|
iter_clear (iter);
|
|
|
|
|
|
Remove assorted G_OBJECT casts where unnecessary.
2001-12-13 Sebastian Wilhelmi <wilhelmi@ira.uka.de>
* gdk-pixbuf-animation.c, gdk-pixbuf-loader.c, gdk-pixpuf.c,
io-gif-animation.c, io-gif.c, io-tiff.c, test-loaders.c: Remove
assorted G_OBJECT casts where unnecessary.
* gdk-pixbuf-loader.c: Call g_object_ref and g_object_unref
instead of gdk_pixbuf_animation_ref and gdk_pixbuf_animation_unref
resp.
* gdk-pixbuf-csource.c, io-bmp.c, io-gif-animation.c, io-ico.c,
io-jpeg.c, io-png.c, io-pnm.c, io-ras.c, io-tga.c, io-wbmp.c,
io-xbm.c, io-xpm.c, test-gdk-pixbuf.c: Dito for gdk_pixbuf_ref and
gdk_pixbuf_unref.
* Makefile.am, pixops/Makefile.am: Compile everything with
-DG_DISABLE_DEPRECATED -DGDK_PIXBUF_DISABLE_DEPRECATED
* gdk-pixdata.c: Use g_ascii_strup() instead of g_strup().
* io-xpm.c: Use g_ascii_strcasecmp() instead of g_strcasecmp().
* demos/testpixbuf-drawable.c, demos/testpixbuf-save.c,
demos/testpixbuf-scale.c, demos/testpixbuf.c: Call g_object_ref
and g_object_unref instead of gdk_pixbuf_ref and gdk_pixbuf_unref
resp.
2001-12-13 21:22:12 +00:00
|
|
|
|
g_object_unref (iter->gif_anim);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (iter_parent_class)->finalize (object);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
gdk_pixbuf_gif_anim_iter_advance (GdkPixbufAnimationIter *anim_iter,
|
|
|
|
|
const GTimeVal *current_time)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnimIter *iter;
|
|
|
|
|
gint elapsed;
|
2002-10-20 22:16:58 +00:00
|
|
|
|
gint loop;
|
2001-05-07 15:58:47 +00:00
|
|
|
|
GList *tmp;
|
|
|
|
|
GList *old;
|
|
|
|
|
|
|
|
|
|
iter = GDK_PIXBUF_GIF_ANIM_ITER (anim_iter);
|
|
|
|
|
|
|
|
|
|
iter->current_time = *current_time;
|
|
|
|
|
|
|
|
|
|
/* We use milliseconds for all times */
|
|
|
|
|
elapsed =
|
|
|
|
|
(((iter->current_time.tv_sec - iter->start_time.tv_sec) * G_USEC_PER_SEC +
|
|
|
|
|
iter->current_time.tv_usec - iter->start_time.tv_usec)) / 1000;
|
2001-06-08 20:15:49 +00:00
|
|
|
|
|
2001-05-07 15:58:47 +00:00
|
|
|
|
if (elapsed < 0) {
|
|
|
|
|
/* Try to compensate; probably the system clock
|
|
|
|
|
* was set backwards
|
|
|
|
|
*/
|
|
|
|
|
iter->start_time = iter->current_time;
|
|
|
|
|
elapsed = 0;
|
|
|
|
|
}
|
2001-06-08 20:15:49 +00:00
|
|
|
|
|
|
|
|
|
g_assert (iter->gif_anim->total_time > 0);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
/* See how many times we've already played the full animation,
|
|
|
|
|
* and subtract time for that.
|
|
|
|
|
*/
|
2002-10-20 22:16:58 +00:00
|
|
|
|
|
2003-05-19 21:44:03 +00:00
|
|
|
|
if (iter->gif_anim->loading)
|
|
|
|
|
loop = 0;
|
|
|
|
|
else {
|
|
|
|
|
/* If current_frame is NULL at this point, we have loaded the
|
|
|
|
|
* animation from a source which fell behind the speed of the
|
|
|
|
|
* display. We remember how much slower the first loop was due
|
|
|
|
|
* to this and correct the position calculation in order to not
|
|
|
|
|
* jump in the middle of the second loop.
|
|
|
|
|
*/
|
|
|
|
|
if (iter->current_frame == NULL)
|
|
|
|
|
iter->first_loop_slowness = MAX(0, elapsed - iter->gif_anim->total_time);
|
|
|
|
|
|
|
|
|
|
loop = (elapsed - iter->first_loop_slowness) / iter->gif_anim->total_time;
|
|
|
|
|
elapsed = (elapsed - iter->first_loop_slowness) % iter->gif_anim->total_time;
|
|
|
|
|
}
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
iter->position = elapsed;
|
2002-10-20 22:16:58 +00:00
|
|
|
|
|
2001-05-07 15:58:47 +00:00
|
|
|
|
/* Now move to the proper frame */
|
2002-10-20 22:16:58 +00:00
|
|
|
|
if (iter->gif_anim->loop == 0 || loop < iter->gif_anim->loop)
|
|
|
|
|
tmp = iter->gif_anim->frames;
|
|
|
|
|
else
|
|
|
|
|
tmp = NULL;
|
2001-05-07 15:58:47 +00:00
|
|
|
|
while (tmp != NULL) {
|
|
|
|
|
GdkPixbufFrame *frame = tmp->data;
|
|
|
|
|
|
|
|
|
|
if (iter->position >= frame->elapsed &&
|
|
|
|
|
iter->position < (frame->elapsed + frame->delay_time))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
tmp = tmp->next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
old = iter->current_frame;
|
|
|
|
|
|
|
|
|
|
iter->current_frame = tmp;
|
|
|
|
|
|
|
|
|
|
return iter->current_frame != old;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
gdk_pixbuf_gif_anim_iter_get_delay_time (GdkPixbufAnimationIter *anim_iter)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufFrame *frame;
|
|
|
|
|
GdkPixbufGifAnimIter *iter;
|
|
|
|
|
|
|
|
|
|
iter = GDK_PIXBUF_GIF_ANIM_ITER (anim_iter);
|
|
|
|
|
|
|
|
|
|
if (iter->current_frame) {
|
|
|
|
|
frame = iter->current_frame->data;
|
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
g_print ("frame start: %d pos: %d frame len: %d frame remaining: %d\n",
|
|
|
|
|
frame->elapsed,
|
|
|
|
|
iter->position,
|
|
|
|
|
frame->delay_time,
|
|
|
|
|
frame->delay_time - (iter->position - frame->elapsed));
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
return frame->delay_time - (iter->position - frame->elapsed);
|
2003-05-19 21:44:03 +00:00
|
|
|
|
} else
|
2001-05-07 15:58:47 +00:00
|
|
|
|
return -1; /* show last frame forever */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
gdk_pixbuf_gif_anim_frame_composite (GdkPixbufGifAnim *gif_anim,
|
|
|
|
|
GdkPixbufFrame *frame)
|
|
|
|
|
{
|
|
|
|
|
GList *link;
|
|
|
|
|
GList *tmp;
|
|
|
|
|
|
|
|
|
|
link = g_list_find (gif_anim->frames, frame);
|
|
|
|
|
|
|
|
|
|
if (frame->need_recomposite || frame->composited == NULL) {
|
|
|
|
|
/* For now, to composite we start with the last
|
|
|
|
|
* composited frame and composite everything up to
|
|
|
|
|
* here.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* Rewind to last composited frame. */
|
|
|
|
|
tmp = link;
|
|
|
|
|
while (tmp != NULL) {
|
|
|
|
|
GdkPixbufFrame *f = tmp->data;
|
|
|
|
|
|
|
|
|
|
if (f->need_recomposite) {
|
|
|
|
|
if (f->composited) {
|
Remove assorted G_OBJECT casts where unnecessary.
2001-12-13 Sebastian Wilhelmi <wilhelmi@ira.uka.de>
* gdk-pixbuf-animation.c, gdk-pixbuf-loader.c, gdk-pixpuf.c,
io-gif-animation.c, io-gif.c, io-tiff.c, test-loaders.c: Remove
assorted G_OBJECT casts where unnecessary.
* gdk-pixbuf-loader.c: Call g_object_ref and g_object_unref
instead of gdk_pixbuf_animation_ref and gdk_pixbuf_animation_unref
resp.
* gdk-pixbuf-csource.c, io-bmp.c, io-gif-animation.c, io-ico.c,
io-jpeg.c, io-png.c, io-pnm.c, io-ras.c, io-tga.c, io-wbmp.c,
io-xbm.c, io-xpm.c, test-gdk-pixbuf.c: Dito for gdk_pixbuf_ref and
gdk_pixbuf_unref.
* Makefile.am, pixops/Makefile.am: Compile everything with
-DG_DISABLE_DEPRECATED -DGDK_PIXBUF_DISABLE_DEPRECATED
* gdk-pixdata.c: Use g_ascii_strup() instead of g_strup().
* io-xpm.c: Use g_ascii_strcasecmp() instead of g_strcasecmp().
* demos/testpixbuf-drawable.c, demos/testpixbuf-save.c,
demos/testpixbuf-scale.c, demos/testpixbuf.c: Call g_object_ref
and g_object_unref instead of gdk_pixbuf_ref and gdk_pixbuf_unref
resp.
2001-12-13 21:22:12 +00:00
|
|
|
|
g_object_unref (f->composited);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
f->composited = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (f->composited != NULL)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
tmp = tmp->prev;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Go forward, compositing all frames up to the current frame */
|
|
|
|
|
if (tmp == NULL)
|
|
|
|
|
tmp = gif_anim->frames;
|
|
|
|
|
|
|
|
|
|
while (tmp != NULL) {
|
|
|
|
|
GdkPixbufFrame *f = tmp->data;
|
2005-03-09 18:20:04 +00:00
|
|
|
|
gint clipped_width, clipped_height;
|
2005-03-03 13:29:01 +00:00
|
|
|
|
|
|
|
|
|
if (f->pixbuf == NULL)
|
|
|
|
|
return;
|
|
|
|
|
|
2005-03-09 18:20:04 +00:00
|
|
|
|
clipped_width = MIN (gif_anim->width - f->x_offset, gdk_pixbuf_get_width (f->pixbuf));
|
|
|
|
|
clipped_height = MIN (gif_anim->height - f->y_offset, gdk_pixbuf_get_height (f->pixbuf));
|
2003-06-22 18:08:33 +00:00
|
|
|
|
|
2001-05-07 15:58:47 +00:00
|
|
|
|
if (f->need_recomposite) {
|
|
|
|
|
if (f->composited) {
|
Remove assorted G_OBJECT casts where unnecessary.
2001-12-13 Sebastian Wilhelmi <wilhelmi@ira.uka.de>
* gdk-pixbuf-animation.c, gdk-pixbuf-loader.c, gdk-pixpuf.c,
io-gif-animation.c, io-gif.c, io-tiff.c, test-loaders.c: Remove
assorted G_OBJECT casts where unnecessary.
* gdk-pixbuf-loader.c: Call g_object_ref and g_object_unref
instead of gdk_pixbuf_animation_ref and gdk_pixbuf_animation_unref
resp.
* gdk-pixbuf-csource.c, io-bmp.c, io-gif-animation.c, io-ico.c,
io-jpeg.c, io-png.c, io-pnm.c, io-ras.c, io-tga.c, io-wbmp.c,
io-xbm.c, io-xpm.c, test-gdk-pixbuf.c: Dito for gdk_pixbuf_ref and
gdk_pixbuf_unref.
* Makefile.am, pixops/Makefile.am: Compile everything with
-DG_DISABLE_DEPRECATED -DGDK_PIXBUF_DISABLE_DEPRECATED
* gdk-pixdata.c: Use g_ascii_strup() instead of g_strup().
* io-xpm.c: Use g_ascii_strcasecmp() instead of g_strcasecmp().
* demos/testpixbuf-drawable.c, demos/testpixbuf-save.c,
demos/testpixbuf-scale.c, demos/testpixbuf.c: Call g_object_ref
and g_object_unref instead of gdk_pixbuf_ref and gdk_pixbuf_unref
resp.
2001-12-13 21:22:12 +00:00
|
|
|
|
g_object_unref (f->composited);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
f->composited = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (f->composited != NULL)
|
|
|
|
|
goto next;
|
|
|
|
|
|
|
|
|
|
if (tmp->prev == NULL) {
|
|
|
|
|
/* First frame may be smaller than the whole image;
|
|
|
|
|
* if so, we make the area outside it full alpha if the
|
|
|
|
|
* image has alpha, and background color otherwise.
|
|
|
|
|
* GIF spec doesn't actually say what to do about this.
|
|
|
|
|
*/
|
|
|
|
|
f->composited = gdk_pixbuf_new (GDK_COLORSPACE_RGB,
|
|
|
|
|
TRUE,
|
|
|
|
|
8, gif_anim->width, gif_anim->height);
|
|
|
|
|
|
2005-03-03 13:29:01 +00:00
|
|
|
|
if (f->composited == NULL)
|
|
|
|
|
return;
|
|
|
|
|
|
2001-05-07 15:58:47 +00:00
|
|
|
|
/* alpha gets dumped if f->composited has no alpha */
|
|
|
|
|
|
|
|
|
|
gdk_pixbuf_fill (f->composited,
|
|
|
|
|
(gif_anim->bg_red << 24) |
|
|
|
|
|
(gif_anim->bg_green << 16) |
|
2003-02-19 23:02:03 +00:00
|
|
|
|
(gif_anim->bg_blue << 8));
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
2003-06-22 18:08:33 +00:00
|
|
|
|
if (clipped_width > 0 && clipped_height > 0)
|
|
|
|
|
gdk_pixbuf_composite (f->pixbuf,
|
|
|
|
|
f->composited,
|
|
|
|
|
f->x_offset,
|
|
|
|
|
f->y_offset,
|
|
|
|
|
clipped_width,
|
|
|
|
|
clipped_height,
|
|
|
|
|
f->x_offset, f->y_offset,
|
|
|
|
|
1.0, 1.0,
|
|
|
|
|
GDK_INTERP_BILINEAR,
|
|
|
|
|
255);
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
if (f->action == GDK_PIXBUF_FRAME_REVERT)
|
|
|
|
|
g_warning ("First frame of GIF has bad dispose mode, GIF loader should not have loaded this image");
|
|
|
|
|
|
|
|
|
|
f->need_recomposite = FALSE;
|
|
|
|
|
} else {
|
|
|
|
|
GdkPixbufFrame *prev_frame;
|
2003-06-27 02:38:44 +00:00
|
|
|
|
gint prev_clipped_width;
|
|
|
|
|
gint prev_clipped_height;
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
prev_frame = tmp->prev->data;
|
|
|
|
|
|
2003-06-27 02:38:44 +00:00
|
|
|
|
prev_clipped_width = MIN (gif_anim->width - prev_frame->x_offset, gdk_pixbuf_get_width (prev_frame->pixbuf));
|
|
|
|
|
prev_clipped_height = MIN (gif_anim->height - prev_frame->y_offset, gdk_pixbuf_get_height (prev_frame->pixbuf));
|
2003-06-22 18:08:33 +00:00
|
|
|
|
|
2001-05-07 15:58:47 +00:00
|
|
|
|
/* Init f->composited with what we should have after the previous
|
|
|
|
|
* frame
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
if (prev_frame->action == GDK_PIXBUF_FRAME_RETAIN) {
|
|
|
|
|
f->composited = gdk_pixbuf_copy (prev_frame->composited);
|
2005-03-03 13:29:01 +00:00
|
|
|
|
|
|
|
|
|
if (f->composited == NULL)
|
|
|
|
|
return;
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
} else if (prev_frame->action == GDK_PIXBUF_FRAME_DISPOSE) {
|
|
|
|
|
f->composited = gdk_pixbuf_copy (prev_frame->composited);
|
2005-03-03 13:29:01 +00:00
|
|
|
|
|
|
|
|
|
if (f->composited == NULL)
|
|
|
|
|
return;
|
|
|
|
|
|
2003-06-22 18:08:33 +00:00
|
|
|
|
if (prev_clipped_width > 0 && prev_clipped_height > 0) {
|
|
|
|
|
/* Clear area of previous frame to background */
|
|
|
|
|
GdkPixbuf *area;
|
|
|
|
|
|
|
|
|
|
area = gdk_pixbuf_new_subpixbuf (f->composited,
|
|
|
|
|
prev_frame->x_offset,
|
|
|
|
|
prev_frame->y_offset,
|
|
|
|
|
prev_clipped_width,
|
|
|
|
|
prev_clipped_height);
|
2005-03-03 13:29:01 +00:00
|
|
|
|
|
|
|
|
|
if (area == NULL)
|
|
|
|
|
return;
|
2003-06-22 18:08:33 +00:00
|
|
|
|
|
|
|
|
|
gdk_pixbuf_fill (area,
|
|
|
|
|
(gif_anim->bg_red << 24) |
|
|
|
|
|
(gif_anim->bg_green << 16) |
|
|
|
|
|
(gif_anim->bg_blue << 8));
|
|
|
|
|
|
|
|
|
|
g_object_unref (area);
|
|
|
|
|
}
|
2001-05-07 15:58:47 +00:00
|
|
|
|
} else if (prev_frame->action == GDK_PIXBUF_FRAME_REVERT) {
|
|
|
|
|
f->composited = gdk_pixbuf_copy (prev_frame->composited);
|
2005-03-03 13:29:01 +00:00
|
|
|
|
|
|
|
|
|
if (f->composited == NULL)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (prev_frame->revert != NULL &&
|
|
|
|
|
prev_clipped_width > 0 && prev_clipped_height > 0) {
|
2003-06-22 18:08:33 +00:00
|
|
|
|
/* Copy in the revert frame */
|
|
|
|
|
gdk_pixbuf_copy_area (prev_frame->revert,
|
|
|
|
|
0, 0,
|
|
|
|
|
gdk_pixbuf_get_width (prev_frame->revert),
|
|
|
|
|
gdk_pixbuf_get_height (prev_frame->revert),
|
|
|
|
|
f->composited,
|
|
|
|
|
prev_frame->x_offset,
|
|
|
|
|
prev_frame->y_offset);
|
|
|
|
|
}
|
2001-05-07 15:58:47 +00:00
|
|
|
|
} else {
|
|
|
|
|
g_warning ("Unknown revert action for GIF frame");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (f->revert == NULL &&
|
|
|
|
|
f->action == GDK_PIXBUF_FRAME_REVERT) {
|
2003-06-22 18:08:33 +00:00
|
|
|
|
if (clipped_width > 0 && clipped_height > 0) {
|
|
|
|
|
/* We need to save the contents before compositing */
|
|
|
|
|
GdkPixbuf *area;
|
|
|
|
|
|
|
|
|
|
area = gdk_pixbuf_new_subpixbuf (f->composited,
|
|
|
|
|
f->x_offset,
|
|
|
|
|
f->y_offset,
|
|
|
|
|
clipped_width,
|
|
|
|
|
clipped_height);
|
2005-03-03 13:29:01 +00:00
|
|
|
|
|
|
|
|
|
if (area == NULL)
|
|
|
|
|
return;
|
2003-06-22 18:08:33 +00:00
|
|
|
|
|
|
|
|
|
f->revert = gdk_pixbuf_copy (area);
|
2005-03-03 13:29:01 +00:00
|
|
|
|
|
2003-06-22 18:08:33 +00:00
|
|
|
|
g_object_unref (area);
|
2005-03-03 13:29:01 +00:00
|
|
|
|
|
|
|
|
|
if (f->revert == NULL)
|
|
|
|
|
return;
|
2003-06-22 18:08:33 +00:00
|
|
|
|
}
|
2001-05-07 15:58:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
2005-03-03 13:29:01 +00:00
|
|
|
|
if (clipped_width > 0 && clipped_height > 0 &&
|
|
|
|
|
f->pixbuf != NULL && f->composited != NULL) {
|
2003-06-22 18:08:33 +00:00
|
|
|
|
/* Put current frame onto f->composited */
|
|
|
|
|
gdk_pixbuf_composite (f->pixbuf,
|
|
|
|
|
f->composited,
|
|
|
|
|
f->x_offset,
|
|
|
|
|
f->y_offset,
|
|
|
|
|
clipped_width,
|
|
|
|
|
clipped_height,
|
|
|
|
|
f->x_offset, f->y_offset,
|
|
|
|
|
1.0, 1.0,
|
|
|
|
|
GDK_INTERP_NEAREST,
|
|
|
|
|
255);
|
|
|
|
|
}
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
f->need_recomposite = FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
next:
|
|
|
|
|
if (tmp == link)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
tmp = tmp->next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
GdkPixbuf*
|
|
|
|
|
gdk_pixbuf_gif_anim_iter_get_pixbuf (GdkPixbufAnimationIter *anim_iter)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnimIter *iter;
|
|
|
|
|
GdkPixbufFrame *frame;
|
|
|
|
|
|
|
|
|
|
iter = GDK_PIXBUF_GIF_ANIM_ITER (anim_iter);
|
|
|
|
|
|
2002-10-20 22:16:58 +00:00
|
|
|
|
frame = iter->current_frame ? iter->current_frame->data : g_list_last (iter->gif_anim->frames)->data;
|
2001-05-07 15:58:47 +00:00
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
if (FALSE && frame)
|
|
|
|
|
g_print ("current frame %d dispose mode %d %d x %d\n",
|
|
|
|
|
g_list_index (iter->gif_anim->frames,
|
|
|
|
|
frame),
|
|
|
|
|
frame->action,
|
|
|
|
|
gdk_pixbuf_get_width (frame->pixbuf),
|
|
|
|
|
gdk_pixbuf_get_height (frame->pixbuf));
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if (frame == NULL)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
gdk_pixbuf_gif_anim_frame_composite (iter->gif_anim, frame);
|
|
|
|
|
|
|
|
|
|
return frame->composited;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
|
gdk_pixbuf_gif_anim_iter_on_currently_loading_frame (GdkPixbufAnimationIter *anim_iter)
|
|
|
|
|
{
|
|
|
|
|
GdkPixbufGifAnimIter *iter;
|
|
|
|
|
|
|
|
|
|
iter = GDK_PIXBUF_GIF_ANIM_ITER (anim_iter);
|
|
|
|
|
|
|
|
|
|
return iter->current_frame == NULL || iter->current_frame->next == NULL;
|
|
|
|
|
}
|