Blame gdk-pixbuf/io-gdip-animation.c

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
}