|
Packit |
a4058c |
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
Packit |
a4058c |
/* GdkPixbuf library - animated gdip support
|
|
Packit |
a4058c |
*
|
|
Packit |
a4058c |
* Copyright (C) 1999 The Free Software Foundation
|
|
Packit |
a4058c |
*
|
|
Packit |
a4058c |
* Authors: Jonathan Blandford <jrb@redhat.com>
|
|
Packit |
a4058c |
* Havoc Pennington <hp@redhat.com>
|
|
Packit |
a4058c |
*
|
|
Packit |
a4058c |
* This library is free software; you can redistribute it and/or
|
|
Packit |
a4058c |
* modify it under the terms of the GNU Lesser General Public
|
|
Packit |
a4058c |
* License as published by the Free Software Foundation; either
|
|
Packit |
a4058c |
* version 2 of the License, or (at your option) any later version.
|
|
Packit |
a4058c |
*
|
|
Packit |
a4058c |
* This library is distributed in the hope that it will be useful,
|
|
Packit |
a4058c |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
a4058c |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Packit |
a4058c |
* Lesser General Public License for more details.
|
|
Packit |
a4058c |
*
|
|
Packit |
a4058c |
* You should have received a copy of the GNU Lesser General Public
|
|
Packit |
a4058c |
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
Packit |
a4058c |
*/
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
#include <errno.h>
|
|
Packit |
a4058c |
#include "io-gdip-native.h"
|
|
Packit |
a4058c |
#include "io-gdip-animation.h"
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void gdk_pixbuf_gdip_anim_finalize (GObject *object);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static gboolean gdk_pixbuf_gdip_anim_is_static_image (GdkPixbufAnimation *animation);
|
|
Packit |
a4058c |
static GdkPixbuf* gdk_pixbuf_gdip_anim_get_static_image (GdkPixbufAnimation *animation);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void gdk_pixbuf_gdip_anim_get_size (GdkPixbufAnimation *anim,
|
|
Packit |
a4058c |
int *width,
|
|
Packit |
a4058c |
int *height);
|
|
Packit |
a4058c |
static GdkPixbufAnimationIter* gdk_pixbuf_gdip_anim_get_iter (GdkPixbufAnimation *anim,
|
|
Packit |
a4058c |
const GTimeVal *start_time);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
G_DEFINE_TYPE (GdkPixbufGdipAnim, gdk_pixbuf_gdip_anim, GDK_TYPE_PIXBUF_ANIMATION);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_init (GdkPixbufGdipAnim *anim)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_class_init (GdkPixbufGdipAnimClass *klass)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
Packit |
a4058c |
GdkPixbufAnimationClass *anim_class = GDK_PIXBUF_ANIMATION_CLASS (klass);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
object_class->finalize = gdk_pixbuf_gdip_anim_finalize;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
anim_class->is_static_image = gdk_pixbuf_gdip_anim_is_static_image;
|
|
Packit |
a4058c |
anim_class->get_static_image = gdk_pixbuf_gdip_anim_get_static_image;
|
|
Packit |
a4058c |
anim_class->get_size = gdk_pixbuf_gdip_anim_get_size;
|
|
Packit |
a4058c |
anim_class->get_iter = gdk_pixbuf_gdip_anim_get_iter;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_finalize (GObject *object)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnim *gdip_anim = GDK_PIXBUF_GDIP_ANIM (object);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
GList *l;
|
|
Packit |
a4058c |
GdkPixbufFrame *frame;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
for (l = gdip_anim->frames; l; l = l->next) {
|
|
Packit |
a4058c |
frame = l->data;
|
|
Packit |
a4058c |
g_object_unref (frame->pixbuf);
|
|
Packit |
a4058c |
g_free (frame);
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
g_list_free (gdip_anim->frames);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
G_OBJECT_CLASS (gdk_pixbuf_gdip_anim_parent_class)->finalize (object);
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static gboolean
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_is_static_image (GdkPixbufAnimation *animation)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnim *gdip_anim;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
gdip_anim = GDK_PIXBUF_GDIP_ANIM (animation);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
return (gdip_anim->frames != NULL &&
|
|
Packit |
a4058c |
gdip_anim->frames->next == NULL);
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static GdkPixbuf*
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_get_static_image (GdkPixbufAnimation *animation)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnim *gdip_anim;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
gdip_anim = GDK_PIXBUF_GDIP_ANIM (animation);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
if (gdip_anim->frames == NULL)
|
|
Packit |
a4058c |
return NULL;
|
|
Packit |
a4058c |
else
|
|
Packit |
a4058c |
return GDK_PIXBUF (((GdkPixbufFrame*)gdip_anim->frames->data)->pixbuf);
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_get_size (GdkPixbufAnimation *anim,
|
|
Packit |
a4058c |
int *width,
|
|
Packit |
a4058c |
int *height)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnim *gdip_anim;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
gdip_anim = GDK_PIXBUF_GDIP_ANIM (anim);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
if (width)
|
|
Packit |
a4058c |
*width = gdip_anim->width;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
if (height)
|
|
Packit |
a4058c |
*height = gdip_anim->height;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
iter_clear (GdkPixbufGdipAnimIter *iter)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
iter->current_frame = NULL;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
iter_restart (GdkPixbufGdipAnimIter *iter)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
iter_clear (iter);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter->current_frame = iter->gdip_anim->frames;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static GdkPixbufAnimationIter*
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_get_iter (GdkPixbufAnimation *anim,
|
|
Packit |
a4058c |
const GTimeVal *start_time)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnimIter *iter;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter = g_object_new (GDK_TYPE_PIXBUF_GDIP_ANIM_ITER, NULL);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter->gdip_anim = GDK_PIXBUF_GDIP_ANIM (anim);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
g_object_ref (iter->gdip_anim);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter_restart (iter);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter->start_time = *start_time;
|
|
Packit |
a4058c |
iter->current_time = *start_time;
|
|
Packit |
a4058c |
iter->first_loop_slowness = 0;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
return GDK_PIXBUF_ANIMATION_ITER (iter);
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void gdk_pixbuf_gdip_anim_iter_finalize (GObject *object);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static int gdk_pixbuf_gdip_anim_iter_get_delay_time (GdkPixbufAnimationIter *iter);
|
|
Packit |
a4058c |
static GdkPixbuf* gdk_pixbuf_gdip_anim_iter_get_pixbuf (GdkPixbufAnimationIter *iter);
|
|
Packit |
a4058c |
static gboolean gdk_pixbuf_gdip_anim_iter_on_currently_loading_frame (GdkPixbufAnimationIter *iter);
|
|
Packit |
a4058c |
static gboolean gdk_pixbuf_gdip_anim_iter_advance (GdkPixbufAnimationIter *iter,
|
|
Packit |
a4058c |
const GTimeVal *current_time);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
G_DEFINE_TYPE (GdkPixbufGdipAnimIter, gdk_pixbuf_gdip_anim_iter, GDK_TYPE_PIXBUF_ANIMATION_ITER);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_iter_init (GdkPixbufGdipAnimIter *iter)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_iter_class_init (GdkPixbufGdipAnimIterClass *klass)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
Packit |
a4058c |
GdkPixbufAnimationIterClass *anim_iter_class =
|
|
Packit |
a4058c |
GDK_PIXBUF_ANIMATION_ITER_CLASS (klass);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
object_class->finalize = gdk_pixbuf_gdip_anim_iter_finalize;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
anim_iter_class->get_delay_time = gdk_pixbuf_gdip_anim_iter_get_delay_time;
|
|
Packit |
a4058c |
anim_iter_class->get_pixbuf = gdk_pixbuf_gdip_anim_iter_get_pixbuf;
|
|
Packit |
a4058c |
anim_iter_class->on_currently_loading_frame = gdk_pixbuf_gdip_anim_iter_on_currently_loading_frame;
|
|
Packit |
a4058c |
anim_iter_class->advance = gdk_pixbuf_gdip_anim_iter_advance;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static void
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_iter_finalize (GObject *object)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnimIter *iter = GDK_PIXBUF_GDIP_ANIM_ITER (object);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter_clear (iter);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
g_object_unref (iter->gdip_anim);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
G_OBJECT_CLASS (gdk_pixbuf_gdip_anim_iter_parent_class)->finalize (object);
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static gboolean
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_iter_advance (GdkPixbufAnimationIter *anim_iter,
|
|
Packit |
a4058c |
const GTimeVal *current_time)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnimIter *iter;
|
|
Packit |
a4058c |
gint elapsed;
|
|
Packit |
a4058c |
gint loop;
|
|
Packit |
a4058c |
GList *tmp;
|
|
Packit |
a4058c |
GList *old;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter = GDK_PIXBUF_GDIP_ANIM_ITER (anim_iter);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter->current_time = *current_time;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
/* We use milliseconds for all times */
|
|
Packit |
a4058c |
elapsed =
|
|
Packit |
a4058c |
(((iter->current_time.tv_sec - iter->start_time.tv_sec) * G_USEC_PER_SEC +
|
|
Packit |
a4058c |
iter->current_time.tv_usec - iter->start_time.tv_usec)) / 1000;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
if (elapsed < 0) {
|
|
Packit |
a4058c |
/* Try to compensate; probably the system clock
|
|
Packit |
a4058c |
* was set backwards
|
|
Packit |
a4058c |
*/
|
|
Packit |
a4058c |
iter->start_time = iter->current_time;
|
|
Packit |
a4058c |
elapsed = 0;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
g_assert (iter->gdip_anim->total_time > 0);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
/* See how many times we've already played the full animation,
|
|
Packit |
a4058c |
* and subtract time for that.
|
|
Packit |
a4058c |
*/
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
if (iter->gdip_anim->loading)
|
|
Packit |
a4058c |
loop = 0;
|
|
Packit |
a4058c |
else {
|
|
Packit |
a4058c |
/* If current_frame is NULL at this point, we have loaded the
|
|
Packit |
a4058c |
* animation from a source which fell behind the speed of the
|
|
Packit |
a4058c |
* display. We remember how much slower the first loop was due
|
|
Packit |
a4058c |
* to this and correct the position calculation in order to not
|
|
Packit |
a4058c |
* jump in the middle of the second loop.
|
|
Packit |
a4058c |
*/
|
|
Packit |
a4058c |
if (iter->current_frame == NULL)
|
|
Packit |
a4058c |
iter->first_loop_slowness = MAX(0, elapsed - iter->gdip_anim->total_time);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
loop = (elapsed - iter->first_loop_slowness) / iter->gdip_anim->total_time;
|
|
Packit |
a4058c |
elapsed = (elapsed - iter->first_loop_slowness) % iter->gdip_anim->total_time;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter->position = elapsed;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
/* Now move to the proper frame */
|
|
Packit |
a4058c |
if (iter->gdip_anim->loop == 0 || loop < iter->gdip_anim->loop)
|
|
Packit |
a4058c |
tmp = iter->gdip_anim->frames;
|
|
Packit |
a4058c |
else
|
|
Packit |
a4058c |
tmp = NULL;
|
|
Packit |
a4058c |
while (tmp != NULL) {
|
|
Packit |
a4058c |
GdkPixbufFrame *frame = tmp->data;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
if (iter->position >= frame->elapsed &&
|
|
Packit |
a4058c |
iter->position < (frame->elapsed + frame->delay_time))
|
|
Packit |
a4058c |
break;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
tmp = tmp->next;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
old = iter->current_frame;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter->current_frame = tmp;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
return iter->current_frame != old;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
int
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_iter_get_delay_time (GdkPixbufAnimationIter *anim_iter)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufFrame *frame;
|
|
Packit |
a4058c |
GdkPixbufGdipAnimIter *iter;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter = GDK_PIXBUF_GDIP_ANIM_ITER (anim_iter);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
if (iter->current_frame) {
|
|
Packit |
a4058c |
frame = iter->current_frame->data;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
#if 0
|
|
Packit |
a4058c |
g_print ("frame start: %d pos: %d frame len: %d frame remaining: %d\n",
|
|
Packit |
a4058c |
frame->elapsed,
|
|
Packit |
a4058c |
iter->position,
|
|
Packit |
a4058c |
frame->delay_time,
|
|
Packit |
a4058c |
frame->delay_time - (iter->position - frame->elapsed));
|
|
Packit |
a4058c |
#endif
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
return frame->delay_time - (iter->position - frame->elapsed);
|
|
Packit |
a4058c |
} else
|
|
Packit |
a4058c |
return -1; /* show last frame forever */
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
GdkPixbuf*
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_iter_get_pixbuf (GdkPixbufAnimationIter *anim_iter)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnimIter *iter;
|
|
Packit |
a4058c |
GdkPixbufFrame *frame;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter = GDK_PIXBUF_GDIP_ANIM_ITER (anim_iter);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
frame = iter->current_frame ? iter->current_frame->data : g_list_last (iter->gdip_anim->frames)->data;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
#if 0
|
|
Packit |
a4058c |
if (FALSE && frame)
|
|
Packit |
a4058c |
g_print ("current frame %d dispose mode %d %d x %d\n",
|
|
Packit |
a4058c |
g_list_index (iter->gdip_anim->frames,
|
|
Packit |
a4058c |
frame),
|
|
Packit |
a4058c |
frame->action,
|
|
Packit |
a4058c |
gdk_pixbuf_get_width (frame->pixbuf),
|
|
Packit |
a4058c |
gdk_pixbuf_get_height (frame->pixbuf));
|
|
Packit |
a4058c |
#endif
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
if (frame == NULL)
|
|
Packit |
a4058c |
return NULL;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
return frame->pixbuf;
|
|
Packit |
a4058c |
}
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
static gboolean
|
|
Packit |
a4058c |
gdk_pixbuf_gdip_anim_iter_on_currently_loading_frame (GdkPixbufAnimationIter *anim_iter)
|
|
Packit |
a4058c |
{
|
|
Packit |
a4058c |
GdkPixbufGdipAnimIter *iter;
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
iter = GDK_PIXBUF_GDIP_ANIM_ITER (anim_iter);
|
|
Packit |
a4058c |
|
|
Packit |
a4058c |
return iter->current_frame == NULL || iter->current_frame->next == NULL;
|
|
Packit |
a4058c |
}
|