[gimp/wip/animation: 80/197] plug-ins: split the playback code out of the Animation class.
- From: Jehan Pagès <jehanp src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gimp/wip/animation: 80/197] plug-ins: split the playback code out of the Animation class.
- Date: Sat, 7 Oct 2017 03:05:07 +0000 (UTC)
commit 7833558f587b910261286c16dbb5f15135eedf89
Author: Jehan <jehan girinstud io>
Date: Fri Nov 4 02:16:38 2016 +0100
plug-ins: split the playback code out of the Animation class.
plug-ins/animation-play/Makefile.am | 2 +
.../animation-play/core/animation-celanimation.c | 13 +-
plug-ins/animation-play/core/animation-playback.c | 628 ++++++++++++++++++++
plug-ins/animation-play/core/animation-playback.h | 86 +++
plug-ins/animation-play/core/animation.c | 512 +++--------------
plug-ins/animation-play/core/animation.h | 46 +-
plug-ins/animation-play/core/animationanimatic.c | 109 +---
plug-ins/animation-play/core/animationanimatic.h | 33 +-
plug-ins/animation-play/widgets/animation-dialog.c | 322 +++++-----
.../animation-play/widgets/animation-storyboard.c | 66 ++-
.../animation-play/widgets/animation-storyboard.h | 3 +-
plug-ins/animation-play/widgets/animation-xsheet.c | 49 +-
plug-ins/animation-play/widgets/animation-xsheet.h | 1 +
13 files changed, 1075 insertions(+), 795 deletions(-)
---
diff --git a/plug-ins/animation-play/Makefile.am b/plug-ins/animation-play/Makefile.am
index 4774dd7..ac66c6e 100644
--- a/plug-ins/animation-play/Makefile.am
+++ b/plug-ins/animation-play/Makefile.am
@@ -49,6 +49,8 @@ animation_play_SOURCES = \
core/animationanimatic.c \
core/animation-celanimation.h \
core/animation-celanimation.c \
+ core/animation-playback.h \
+ core/animation-playback.c \
widgets/animation-dialog.h \
widgets/animation-dialog.c \
widgets/animation-storyboard.h \
diff --git a/plug-ins/animation-play/core/animation-celanimation.c
b/plug-ins/animation-play/core/animation-celanimation.c
index 703605c..d049c2d 100644
--- a/plug-ins/animation-play/core/animation-celanimation.c
+++ b/plug-ins/animation-play/core/animation-celanimation.c
@@ -220,16 +220,6 @@ animation_cel_animation_set_layers (AnimationCelAnimation *animation,
frame_layer->data = 0;
}
animation_cel_animation_cache (animation, position);
- if (animation_get_position (ANIMATION (animation)) == position)
- {
- GeglBuffer *buffer;
-
- buffer = animation_get_frame (ANIMATION (animation), position);
- g_signal_emit_by_name (animation, "render",
- position, buffer, TRUE);
- if (buffer)
- g_object_unref (buffer);
- }
}
}
@@ -1088,8 +1078,9 @@ animation_cel_animation_cache (AnimationCelAnimation *animation,
/* This item exists and has a NULL data. */
g_list_nth (animation->priv->cache, pos)->data = cache;
-}
+ g_signal_emit_by_name (animation, "cache-invalidated", pos, 1);
+}
static gboolean
animation_cel_animation_cache_cmp (Cache *cache1,
diff --git a/plug-ins/animation-play/core/animation-playback.c
b/plug-ins/animation-play/core/animation-playback.c
new file mode 100644
index 0000000..e2e6321
--- /dev/null
+++ b/plug-ins/animation-play/core/animation-playback.c
@@ -0,0 +1,628 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * animation-playback.c
+ * Copyright (C) 2016 Jehan <jehan 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 3 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include <string.h>
+
+#include <libgimp/gimp.h>
+
+#include "animation.h"
+#include "animation-playback.h"
+
+enum
+{
+ START,
+ STOP,
+ RANGE,
+ RENDER,
+ LOW_FRAMERATE,
+ LAST_SIGNAL
+};
+
+enum
+{
+ PROP_0,
+ PROP_ANIMATION
+};
+
+struct _AnimationPlaybackPrivate
+{
+ Animation *animation;
+
+ /* State of the currently loaded playback. */
+ gint position;
+ /* Playback can be a subset of frames. */
+ guint start;
+ guint stop;
+
+ guint timer;
+ gint64 start_time;
+ gint64 frames_played;
+};
+
+
+#define ANIMATION_PLAYBACK_GET_PRIVATE(playback) \
+ G_TYPE_INSTANCE_GET_PRIVATE (playback, \
+ ANIMATION_PLAYBACK_TYPE_ANIMATION_PLAYBACK, \
+ AnimationPlaybackPrivate)
+
+static void animation_playback_finalize (GObject *object);
+static void animation_playback_set_property (GObject *object,
+ guint property_id,
+ const GValue *value,
+ GParamSpec *pspec);
+static void animation_playback_get_property (GObject *object,
+ guint property_id,
+ GValue *value,
+ GParamSpec *pspec);
+
+static void on_cache_invalidated (Animation *animation,
+ gint position,
+ gint length,
+ AnimationPlayback *playback);
+
+/* Timer callback for playback. */
+static gboolean animation_playback_advance_frame_callback (AnimationPlayback *playback);
+
+
+G_DEFINE_TYPE (AnimationPlayback, animation_playback, G_TYPE_OBJECT)
+
+#define parent_class animation_playback_parent_class
+
+static guint animation_playback_signals[LAST_SIGNAL] = { 0 };
+
+static void
+animation_playback_class_init (AnimationPlaybackClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ /**
+ * AnimationPlayback::start:
+ * @playback: the #AnimationPlayback.
+ *
+ * The @playback starts to play.
+ */
+ animation_playback_signals[START] =
+ g_signal_new ("start",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (AnimationPlaybackClass, start),
+ NULL, NULL,
+ g_cclosure_marshal_VOID__VOID,
+ G_TYPE_NONE,
+ 0);
+ /**
+ * AnimationPlayback::stop:
+ * @playback: the #AnimationPlayback.
+ *
+ * The @playback stops playing.
+ */
+ animation_playback_signals[STOP] =
+ g_signal_new ("stop",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (AnimationPlaybackClass, stop),
+ NULL, NULL,
+ g_cclosure_marshal_VOID__VOID,
+ G_TYPE_NONE,
+ 0);
+ /**
+ * AnimationPlayback::range:
+ * @playback: the #AnimationPlayback.
+ * @start: the playback start frame.
+ * @stop: the playback last frame.
+ *
+ * The ::range signal is emitted when the playback range is
+ * updated.
+ */
+ animation_playback_signals[RANGE] =
+ g_signal_new ("range",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (AnimationPlaybackClass, range),
+ NULL, NULL,
+ NULL,
+ G_TYPE_NONE,
+ 2,
+ G_TYPE_INT,
+ G_TYPE_INT);
+ /**
+ * AnimationPlayback::render:
+ * @playback: the #AnimationPlayback.
+ * @position: current position to be rendered.
+ * @buffer: the #GeglBuffer for the frame at @position.
+ * @must_draw_null: meaning of a %NULL @buffer.
+ * %TRUE means we have to draw an empty frame.
+ * %FALSE means the new frame is same as the current frame.
+ *
+ * Sends a request for render to the GUI.
+ */
+ animation_playback_signals[RENDER] =
+ g_signal_new ("render",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (AnimationPlaybackClass, render),
+ NULL, NULL,
+ NULL,
+ G_TYPE_NONE,
+ 3,
+ G_TYPE_INT,
+ GEGL_TYPE_BUFFER,
+ G_TYPE_BOOLEAN);
+ /**
+ * AnimationPlayback::low-framerate:
+ * @playback: the #AnimationPlayback.
+ * @actual_fps: the current playback framerate in fps.
+ *
+ * The ::low-framerate signal is emitted when the playback framerate
+ * is lower than expected. It is also emitted once when the framerate
+ * comes back to acceptable rate.
+ */
+ animation_playback_signals[LOW_FRAMERATE] =
+ g_signal_new ("low-framerate",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (AnimationPlaybackClass, low_framerate),
+ NULL, NULL,
+ g_cclosure_marshal_VOID__DOUBLE,
+ G_TYPE_NONE,
+ 1,
+ G_TYPE_DOUBLE);
+
+ object_class->finalize = animation_playback_finalize;
+ object_class->set_property = animation_playback_set_property;
+ object_class->get_property = animation_playback_get_property;
+
+ /**
+ * AnimationPlayback:animation:
+ *
+ * The associated #Animation.
+ */
+ g_object_class_install_property (object_class, PROP_ANIMATION,
+ g_param_spec_object ("animation",
+ NULL, NULL,
+ ANIMATION_TYPE_ANIMATION,
+ G_PARAM_READWRITE));
+
+ g_type_class_add_private (klass, sizeof (AnimationPlaybackPrivate));
+}
+
+static void
+animation_playback_init (AnimationPlayback *playback)
+{
+ playback->priv = G_TYPE_INSTANCE_GET_PRIVATE (playback,
+ ANIMATION_TYPE_PLAYBACK,
+ AnimationPlaybackPrivate);
+}
+
+/************ Public Functions ****************/
+
+AnimationPlayback *
+animation_playback_new (void)
+{
+ AnimationPlayback *playback;
+
+ playback = g_object_new (ANIMATION_TYPE_PLAYBACK,
+ NULL);
+
+ return playback;
+}
+
+void
+animation_playback_set_animation (AnimationPlayback *playback,
+ Animation *animation)
+{
+ g_object_set (playback,
+ "animation", animation,
+ NULL);
+}
+
+Animation *
+animation_playback_get_animation (AnimationPlayback *playback)
+{
+ return playback->priv->animation;
+}
+
+gint
+animation_playback_get_position (AnimationPlayback *playback)
+{
+ return playback->priv->position;
+}
+
+gboolean
+animation_playback_is_playing (AnimationPlayback *playback)
+{
+ return (playback->priv->timer != 0);
+}
+
+void
+animation_playback_play (AnimationPlayback *playback)
+{
+ gint duration;
+
+ duration = (gint) (1000.0 /
+ animation_get_framerate (playback->priv->animation));
+
+ if (playback->priv->timer)
+ {
+ /* It means we are already playing, so we should not need to play
+ * again.
+ * Still be liberal and simply remove the timer before creating a
+ * new one. */
+ g_source_remove (playback->priv->timer);
+ g_signal_emit (playback, animation_playback_signals[STOP], 0);
+ }
+
+ playback->priv->start_time = g_get_monotonic_time ();
+ playback->priv->frames_played = 1;
+
+ playback->priv->timer = g_timeout_add ((guint) duration,
+ (GSourceFunc) animation_playback_advance_frame_callback,
+ playback);
+ g_signal_emit (playback, animation_playback_signals[START], 0);
+}
+
+void
+animation_playback_stop (AnimationPlayback *playback)
+{
+ if (playback->priv->timer)
+ {
+ /* Stop playing by removing any playback timer. */
+ g_source_remove (playback->priv->timer);
+ playback->priv->timer = 0;
+ g_signal_emit (playback, animation_playback_signals[STOP], 0);
+ }
+}
+
+void
+animation_playback_next (AnimationPlayback *playback)
+{
+ Animation *animation = playback->priv->animation;
+ GeglBuffer *buffer = NULL;
+ gint previous_pos = playback->priv->position;
+ gboolean identical;
+
+ if (! playback->priv->animation)
+ return;
+
+ playback->priv->position = playback->priv->start +
+ ((playback->priv->position - playback->priv->start + 1) %
+ (playback->priv->stop - playback->priv->start + 1));
+
+ identical = ANIMATION_GET_CLASS (animation)->same (animation,
+ previous_pos,
+ playback->priv->position);
+ if (! identical)
+ {
+ buffer = animation_get_frame (animation, playback->priv->position);
+ }
+ g_signal_emit (playback, animation_playback_signals[RENDER], 0,
+ playback->priv->position, buffer, ! identical);
+ if (buffer != NULL)
+ {
+ g_object_unref (buffer);
+ }
+}
+
+void
+animation_playback_prev (AnimationPlayback *playback)
+{
+ Animation *animation = playback->priv->animation;
+ GeglBuffer *buffer = NULL;
+ gint prev_pos = playback->priv->position;
+ gboolean identical;
+
+ if (! playback->priv->animation)
+ return;
+
+ if (playback->priv->position == playback->priv->start)
+ {
+ playback->priv->position = animation_playback_get_stop (playback);
+ }
+ else
+ {
+ --playback->priv->position;
+ }
+
+ identical = ANIMATION_GET_CLASS (animation)->same (animation,
+ prev_pos,
+ playback->priv->position);
+ if (! identical)
+ {
+ buffer = animation_get_frame (animation, playback->priv->position);
+ }
+ g_signal_emit (playback, animation_playback_signals[RENDER], 0,
+ playback->priv->position, buffer, ! identical);
+ if (buffer)
+ g_object_unref (buffer);
+}
+
+void
+animation_playback_jump (AnimationPlayback *playback,
+ gint index)
+{
+ Animation *animation = playback->priv->animation;
+ GeglBuffer *buffer = NULL;
+ gint prev_pos = playback->priv->position;
+ gboolean identical;
+
+ if (! playback->priv->animation)
+ return;
+
+ if (index < playback->priv->start ||
+ index > playback->priv->stop)
+ return;
+ else
+ playback->priv->position = index;
+
+ identical = ANIMATION_GET_CLASS (animation)->same (animation,
+ prev_pos,
+ playback->priv->position);
+ if (! identical)
+ {
+ buffer = animation_get_frame (animation, playback->priv->position);
+ }
+ g_signal_emit (playback, animation_playback_signals[RENDER], 0,
+ playback->priv->position, buffer, ! identical);
+ if (buffer)
+ g_object_unref (buffer);
+}
+
+void
+animation_playback_set_start (AnimationPlayback *playback,
+ gint index)
+{
+ gint duration;
+
+ if (! playback->priv->animation)
+ return;
+
+ duration = animation_get_duration (playback->priv->animation);
+
+ if (index < 0 ||
+ index >= duration)
+ {
+ playback->priv->start = 0;
+ }
+ else
+ {
+ playback->priv->start = index;
+ }
+ if (playback->priv->stop < playback->priv->start)
+ {
+ playback->priv->stop = duration - 1;
+ }
+
+ g_signal_emit (playback, animation_playback_signals[RANGE], 0,
+ playback->priv->start, playback->priv->stop);
+
+ if (playback->priv->position < playback->priv->start ||
+ playback->priv->position > playback->priv->stop)
+ {
+ animation_playback_jump (playback, playback->priv->start);
+ }
+}
+
+gint
+animation_playback_get_start (AnimationPlayback *playback)
+{
+ return playback->priv->start;
+}
+
+void
+animation_playback_set_stop (AnimationPlayback *playback,
+ gint index)
+{
+ gint duration;
+
+ if (! playback->priv->animation)
+ return;
+
+ duration = animation_get_duration (playback->priv->animation);
+
+ if (index < 0 ||
+ index >= duration)
+ {
+ playback->priv->stop = duration - 1;
+ }
+ else
+ {
+ playback->priv->stop = index;
+ }
+ if (playback->priv->stop < playback->priv->start)
+ {
+ playback->priv->start = 0;
+ }
+ g_signal_emit (playback, animation_playback_signals[RANGE], 0,
+ playback->priv->start, playback->priv->stop);
+
+ if (playback->priv->position < playback->priv->start ||
+ playback->priv->position > playback->priv->stop)
+ {
+ animation_playback_jump (playback, playback->priv->start);
+ }
+}
+
+gint
+animation_playback_get_stop (AnimationPlayback *playback)
+{
+ return playback->priv->stop;
+}
+
+/************ Private Functions ****************/
+
+static void
+animation_playback_finalize (GObject *object)
+{
+ AnimationPlayback *playback = ANIMATION_PLAYBACK (object);
+
+ if (playback->priv->animation)
+ g_object_unref (playback->priv->animation);
+
+ G_OBJECT_CLASS (parent_class)->finalize (object);
+}
+
+static void
+animation_playback_set_property (GObject *object,
+ guint property_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ AnimationPlayback *playback = ANIMATION_PLAYBACK (object);
+
+ switch (property_id)
+ {
+ case PROP_ANIMATION:
+ {
+ Animation *animation;
+ GeglBuffer *buffer;
+
+ if (playback->priv->animation)
+ g_object_unref (playback->priv->animation);
+
+ animation = g_value_dup_object (value);
+ playback->priv->animation = animation;
+
+ if (! animation)
+ break;
+
+ if (! animation_loaded (animation))
+ animation_load (animation);
+ /* Default playback is the full range of frames. */
+ playback->priv->position = 0;
+ playback->priv->start = 0;
+ playback->priv->stop = animation_get_duration (animation) - 1;
+
+ g_signal_emit_by_name (animation, "loaded");
+ g_signal_connect (animation, "cache-invalidated",
+ G_CALLBACK (on_cache_invalidated), playback);
+
+ /* Once loaded, let's ask render. */
+ buffer = animation_get_frame (animation, playback->priv->position);
+ g_signal_emit (playback, animation_playback_signals[RENDER], 0,
+ playback->priv->position, buffer, TRUE);
+
+ if (buffer)
+ g_object_unref (buffer);
+ }
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
+ break;
+ }
+}
+
+static void
+animation_playback_get_property (GObject *object,
+ guint property_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ AnimationPlayback *playback = ANIMATION_PLAYBACK (object);
+
+ switch (property_id)
+ {
+ case PROP_ANIMATION:
+ g_value_set_object (value, playback->priv->animation);
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
+ break;
+ }
+}
+
+static void
+on_cache_invalidated (Animation *animation,
+ gint position,
+ gint length,
+ AnimationPlayback *playback)
+{
+ gint cur_position;
+
+ cur_position = animation_playback_get_position (playback);
+
+ if (cur_position >= position &&
+ cur_position < position + length)
+ {
+ GeglBuffer *buffer;
+
+ buffer = animation_get_frame (animation, cur_position);
+ g_signal_emit_by_name (playback, "render",
+ cur_position, buffer, TRUE);
+ if (buffer)
+ {
+ g_object_unref (buffer);
+ }
+ }
+}
+
+static gboolean
+animation_playback_advance_frame_callback (AnimationPlayback *playback)
+{
+ gdouble framerate;
+ gint64 duration;
+ gint64 duration_since_start;
+ static gboolean prev_low_framerate = FALSE;
+
+ framerate = animation_get_framerate (playback->priv->animation);
+
+ animation_playback_next (playback);
+ duration = (gint) (1000.0 / framerate);
+
+ duration_since_start = (g_get_monotonic_time () - playback->priv->start_time) / 1000;
+ duration = duration - (duration_since_start - playback->priv->frames_played * duration);
+
+ if (duration < 1)
+ {
+ if (prev_low_framerate)
+ {
+ /* Let's only warn the user for several subsequent slow frames. */
+ gdouble real_framerate;
+
+ real_framerate = (gdouble) playback->priv->frames_played * 1000.0 / duration_since_start;
+
+ if (real_framerate < framerate)
+ g_signal_emit (playback, animation_playback_signals[LOW_FRAMERATE], 0,
+ real_framerate);
+ }
+ duration = 1;
+ prev_low_framerate = TRUE;
+ }
+ else
+ {
+ if (prev_low_framerate)
+ {
+ /* Let's reset framerate warning. */
+ g_signal_emit (playback, animation_playback_signals[LOW_FRAMERATE], 0,
+ framerate);
+ }
+ prev_low_framerate = FALSE;
+ }
+ playback->priv->frames_played++;
+
+ playback->priv->timer = g_timeout_add ((guint) duration,
+ (GSourceFunc) animation_playback_advance_frame_callback,
+ (AnimationPlayback *) playback);
+
+ return G_SOURCE_REMOVE;
+}
diff --git a/plug-ins/animation-play/core/animation-playback.h
b/plug-ins/animation-play/core/animation-playback.h
new file mode 100644
index 0000000..e7ba300
--- /dev/null
+++ b/plug-ins/animation-play/core/animation-playback.h
@@ -0,0 +1,86 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * animation_playback.h
+ * Copyright (C) 2016 Jehan <jehan 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 3 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef __ANIMATION_PLAYBACK_H__
+#define __ANIMATION_PLAYBACK_H__
+
+#define ANIMATION_TYPE_PLAYBACK (animation_playback_get_type ())
+#define ANIMATION_PLAYBACK(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), ANIMATION_TYPE_PLAYBACK,
AnimationPlayback))
+#define ANIMATION_PLAYBACK_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), ANIMATION_TYPE_PLAYBACK,
AnimationPlaybackClass))
+#define ANIMATION_IS_PLAYBACK(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), ANIMATION_TYPE_PLAYBACK))
+#define ANIMATION_IS_PLAYBACK_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), ANIMATION_TYPE_PLAYBACK))
+#define ANIMATION_PLAYBACK_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), ANIMATION_TYPE_PLAYBACK,
AnimationPlaybackClass))
+
+typedef struct _AnimationPlayback AnimationPlayback;
+typedef struct _AnimationPlaybackClass AnimationPlaybackClass;
+typedef struct _AnimationPlaybackPrivate AnimationPlaybackPrivate;
+
+struct _AnimationPlayback
+{
+ GObject parent_instance;
+
+ AnimationPlaybackPrivate *priv;
+};
+
+struct _AnimationPlaybackClass
+{
+ GObjectClass parent_class;
+
+ /* Signals */
+ void (*start) (AnimationPlayback *playback);
+ void (*stop) (AnimationPlayback *playback);
+ void (*range) (AnimationPlayback *playback,
+ gint start,
+ gint stop);
+ void (*render) (AnimationPlayback *playback,
+ gint position,
+ GeglBuffer *buffer,
+ gboolean must_draw_null);
+ void (*low_framerate) (AnimationPlayback *playback,
+ gdouble actual_fps);
+};
+
+GType animation_playback_get_type (void);
+
+AnimationPlayback * animation_playback_new (void);
+
+void animation_playback_set_animation (AnimationPlayback *playback,
+ Animation *animation);
+Animation * animation_playback_get_animation (AnimationPlayback *playback);
+
+gint animation_playback_get_position (AnimationPlayback *playback);
+
+gboolean animation_playback_is_playing (AnimationPlayback *playback);
+void animation_playback_play (AnimationPlayback *playback);
+void animation_playback_stop (AnimationPlayback *playback);
+void animation_playback_next (AnimationPlayback *playback);
+void animation_playback_prev (AnimationPlayback *playback);
+void animation_playback_jump (AnimationPlayback *playback,
+ gint index);
+
+void animation_playback_set_start (AnimationPlayback *playback,
+ gint index);
+gint animation_playback_get_start (AnimationPlayback *playback);
+
+void animation_playback_set_stop (AnimationPlayback *playback,
+ gint index);
+gint animation_playback_get_stop (AnimationPlayback *playback);
+
+#endif /* __ANIMATION_PLAYBACK_H__ */
diff --git a/plug-ins/animation-play/core/animation.c b/plug-ins/animation-play/core/animation.c
index ad7a8cc..0f6c8f1 100644
--- a/plug-ins/animation-play/core/animation.c
+++ b/plug-ins/animation-play/core/animation.c
@@ -42,16 +42,12 @@ CachedSettings;
enum
{
- LOADING_START,
LOADING,
LOADED,
- START,
- STOP,
+ CACHE_INVALIDATED,
+ DURATION_CHANGED,
FRAMERATE_CHANGED,
- PLAYBACK_RANGE,
- RENDER,
- LOW_FRAMERATE,
- PROXY,
+ PROXY_CHANGED,
LAST_SIGNAL
};
@@ -65,26 +61,15 @@ typedef struct _AnimationPrivate AnimationPrivate;
struct _AnimationPrivate
{
- gint32 image_id;
- gchar *xml;
+ gint32 image_id;
+ gchar *xml;
- gdouble framerate;
-
- guint playback_timer;
-
- /* State of the currently loaded animation. */
- gint position;
- /* Playback can be a subset of frames. */
- guint playback_start;
- guint playback_stop;
+ gdouble framerate;
/* Proxy settings generates a reload. */
- gdouble proxy_ratio;
+ gdouble proxy_ratio;
- gboolean loaded;
-
- gint64 playback_start_time;
- gint64 frames_played;
+ gboolean loaded;
};
@@ -93,6 +78,7 @@ struct _AnimationPrivate
ANIMATION_TYPE_ANIMATION, \
AnimationPrivate)
+static void animation_finalize (GObject *object);
static void animation_set_property (GObject *object,
guint property_id,
const GValue *value,
@@ -103,12 +89,9 @@ static void animation_get_property (GObject *object,
GParamSpec *pspec);
/* Base implementation of virtual methods. */
-static gboolean animation_real_same (Animation *animation,
- gint prev_pos,
- gint next_pos);
-
-/* Timer callback for playback. */
-static gboolean animation_advance_frame_callback (Animation *animation);
+static gboolean animation_real_same (Animation *animation,
+ gint prev_pos,
+ gint next_pos);
G_DEFINE_TYPE (Animation, animation, G_TYPE_OBJECT)
@@ -122,37 +105,22 @@ animation_class_init (AnimationClass *klass)
GObjectClass *object_class = G_OBJECT_CLASS (klass);
/**
- * Animation::loading-start:
- * @animation: the animation loading.
- *
- * The ::loading-start signal is emitted when loading starts.
- * GUI widgets depending on a consistent state of @animation should
- * become unresponsive.
- */
- animation_signals[LOADING_START] =
- g_signal_new ("loading-start",
- G_TYPE_FROM_CLASS (klass),
- G_SIGNAL_RUN_FIRST,
- 0,
- NULL, NULL,
- NULL,
- G_TYPE_NONE,
- 0);
- /**
* Animation::loading:
* @animation: the animation loading.
* @ratio: fraction loaded [0-1].
*
- * The ::loading signal must be emitted by subclass of Animation.
+ * The ::loading signal must be emitted by a subclass of Animation.
* It can be used by a GUI to display a progress bar during loading.
+ * GUI widgets depending on a consistent state of @animation should
+ * become unresponsive.
*/
animation_signals[LOADING] =
g_signal_new ("loading",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
- 0,
+ G_STRUCT_OFFSET (AnimationClass, loading),
NULL, NULL,
- NULL,
+ g_cclosure_marshal_VOID__DOUBLE,
G_TYPE_NONE,
1,
G_TYPE_DOUBLE);
@@ -160,8 +128,6 @@ animation_class_init (AnimationClass *klass)
* Animation::loaded:
* @animation: the animation loading.
* @duration: number of frames.
- * @playback_start: the playback start frame.
- * @playback_stop: the playback last frame.
* @width: display width in pixels.
* @height: display height in pixels.
*
@@ -173,125 +139,64 @@ animation_class_init (AnimationClass *klass)
g_signal_new ("loaded",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
- 0,
+ G_STRUCT_OFFSET (AnimationClass, loaded),
NULL, NULL,
- NULL,
+ g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE,
- 5,
- G_TYPE_INT,
- G_TYPE_INT,
- G_TYPE_INT,
- G_TYPE_INT,
- G_TYPE_INT);
+ 0);
/**
- * Animation::render:
+ * Animation::cache-invalidated:
* @animation: the animation.
- * @position: current position to be rendered.
- * @buffer: the #GeglBuffer for the frame at @position.
- * @must_draw_null: meaning of a %NULL @buffer.
- * %TRUE means we have to draw an empty frame.
- * %FALSE means the new frame is same as the current frame.
+ * @position: the first frame position whose cache is invalid.
+ * @length: the number of invalidated frames from @position.
*
- * Sends a request for render to the GUI.
+ * The ::cache-invalidated signal must be emitted when the contents
+ * of one or more successive frames change.
*/
- animation_signals[RENDER] =
- g_signal_new ("render",
+ animation_signals[CACHE_INVALIDATED] =
+ g_signal_new ("cache-invalidated",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
- 0,
+ G_STRUCT_OFFSET (AnimationClass, cache_invalidated),
NULL, NULL,
NULL,
G_TYPE_NONE,
- 3,
+ 2,
G_TYPE_INT,
- GEGL_TYPE_BUFFER,
- G_TYPE_BOOLEAN);
- /**
- * Animation::start:
- * @animation: the animation.
- *
- * The @animation starts to play.
- */
- animation_signals[START] =
- g_signal_new ("start",
- G_TYPE_FROM_CLASS (klass),
- G_SIGNAL_RUN_FIRST,
- 0,
- NULL, NULL,
- NULL,
- G_TYPE_NONE,
- 0);
+ G_TYPE_INT);
/**
- * Animation::stops:
+ * Animation::duration:
* @animation: the animation.
+ * @duration: the new duration of @animation in number of frames.
*
- * The @animation stops playing.
+ * The ::playback-range signal must be emitted when the duration of
+ * @animation changes.
*/
- animation_signals[STOP] =
- g_signal_new ("stop",
+ animation_signals[DURATION_CHANGED] =
+ g_signal_new ("duration-changed",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
- 0,
+ G_STRUCT_OFFSET (AnimationClass, duration_changed),
NULL, NULL,
- NULL,
+ g_cclosure_marshal_VOID__INT,
G_TYPE_NONE,
- 0);
+ 1,
+ G_TYPE_INT);
/**
* Animation::framerate-changed:
* @animation: the animation.
- * @framerate: the new framerate in frames per second.
+ * @framerate: the new framerate of @animation in frames per second.
*
- * The ::framerate-changed signal is emitted when framerate has
- * changed.
+ * The ::playback-range signal is emitted when the framerate of
+ * @animation changes.
*/
animation_signals[FRAMERATE_CHANGED] =
g_signal_new ("framerate-changed",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
- 0,
- NULL, NULL,
- NULL,
- G_TYPE_NONE,
- 1,
- G_TYPE_DOUBLE);
- /**
- * Animation::playback-range:
- * @animation: the animation.
- * @playback_start: the playback start frame.
- * @playback_stop: the playback last frame.
- * @playback_duration: the playback duration (in frames).
- *
- * The ::playback-range signal is emitted when the playback range is
- * updated.
- */
- animation_signals[PLAYBACK_RANGE] =
- g_signal_new ("playback-range",
- G_TYPE_FROM_CLASS (klass),
- G_SIGNAL_RUN_FIRST,
- 0,
- NULL, NULL,
- NULL,
- G_TYPE_NONE,
- 3,
- G_TYPE_INT,
- G_TYPE_INT,
- G_TYPE_INT);
- /**
- * Animation::low-framerate:
- * @animation: the animation.
- * @actual_fps: the current playback framerate in fps.
- *
- * The ::low-framerate signal is emitted when the playback framerate
- * is lower than expected. It is also emitted once when the framerate
- * comes back to acceptable rate.
- */
- animation_signals[LOW_FRAMERATE] =
- g_signal_new ("low-framerate-playback",
- G_TYPE_FROM_CLASS (klass),
- G_SIGNAL_RUN_FIRST,
- 0,
+ G_STRUCT_OFFSET (AnimationClass, framerate_changed),
NULL, NULL,
- NULL,
+ g_cclosure_marshal_VOID__DOUBLE,
G_TYPE_NONE,
1,
G_TYPE_DOUBLE);
@@ -302,17 +207,18 @@ animation_class_init (AnimationClass *klass)
*
* The ::proxy signal is emitted to announce a change of proxy size.
*/
- animation_signals[PROXY] =
- g_signal_new ("proxy",
+ animation_signals[PROXY_CHANGED] =
+ g_signal_new ("proxy-changed",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_FIRST,
- 0,
+ G_STRUCT_OFFSET (AnimationClass, proxy),
NULL, NULL,
- NULL,
+ g_cclosure_marshal_VOID__DOUBLE,
G_TYPE_NONE,
1,
G_TYPE_DOUBLE);
+ object_class->finalize = animation_finalize;
object_class->set_property = animation_set_property;
object_class->get_property = animation_get_property;
@@ -345,7 +251,7 @@ animation_init (Animation *animation)
gimp_get_data (PLUG_IN_PROC, &settings);
/* Acceptable settings for the default. */
- priv->framerate = settings.framerate; /* fps */
+ priv->framerate = settings.framerate;
priv->proxy_ratio = 1.0;
}
@@ -370,7 +276,7 @@ animation_new (gint32 image_id,
}
gint32
-animation_get_image_id (Animation *animation)
+animation_get_image_id (Animation *animation)
{
AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
@@ -381,47 +287,25 @@ void
animation_load (Animation *animation)
{
AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
- GeglBuffer *buffer;
- gint width, height;
- gint duration;
priv->loaded = FALSE;
- g_signal_emit (animation, animation_signals[LOADING_START], 0);
+ g_signal_emit (animation, animation_signals[LOADING], 0, 0.0);
if (priv->xml)
ANIMATION_GET_CLASS (animation)->load_xml (animation,
priv->xml);
else
ANIMATION_GET_CLASS (animation)->load (animation);
+
/* XML is only used for the first load.
* Any next loads will use internal data. */
g_free (priv->xml);
priv->xml = NULL;
- duration = ANIMATION_GET_CLASS (animation)->get_duration (animation);
- priv->position = 0;
-
- /* Default playback is the full range of frames. */
- priv->playback_start = 0;
- priv->playback_stop = duration - 1;
-
priv->loaded = TRUE;
- animation_get_size (animation, &width, &height);
- g_signal_emit (animation, animation_signals[LOADED], 0,
- duration,
- priv->playback_start,
- priv->playback_stop,
- width,
- height);
-
- /* Once loaded, let's ask render. */
- buffer = animation_get_frame (animation, priv->position);
- g_signal_emit (animation, animation_signals[RENDER], 0,
- priv->position, buffer, TRUE);
-
- if (buffer)
- g_object_unref (buffer);
+ /* XXX */
+ /*g_signal_emit (animation, animation_signals[LOADED], 0);*/
}
void
@@ -502,14 +386,6 @@ animation_save_to_parasite (Animation *animation)
}
gint
-animation_get_position (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
-
- return priv->position;
-}
-
-gint
animation_get_duration (Animation *animation)
{
return ANIMATION_GET_CLASS (animation)->get_duration (animation);
@@ -526,7 +402,7 @@ animation_set_proxy (Animation *animation,
if (priv->proxy_ratio != ratio)
{
priv->proxy_ratio = ratio;
- g_signal_emit (animation, animation_signals[PROXY], 0, ratio);
+ g_signal_emit (animation, animation_signals[PROXY_CHANGED], 0, ratio);
/* A proxy change implies a reload. */
animation_load (animation);
@@ -569,222 +445,6 @@ animation_get_frame (Animation *animation,
return ANIMATION_GET_CLASS (animation)->get_frame (animation, pos);
}
-gboolean
-animation_is_playing (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
-
- return (priv->playback_timer != 0);
-}
-
-void
-animation_play (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
- gint duration;
-
- duration = (gint) (1000.0 / animation_get_framerate (animation));
-
- if (priv->playback_timer)
- {
- /* It means we are already playing, so we should not need to play
- * again.
- * Still be liberal and simply remove the timer before creating a
- * new one. */
- g_source_remove (priv->playback_timer);
- }
-
- priv->playback_start_time = g_get_monotonic_time ();
- priv->frames_played = 1;
-
- priv->playback_timer = g_timeout_add ((guint) duration,
- (GSourceFunc) animation_advance_frame_callback,
- animation);
- g_signal_emit (animation, animation_signals[START], 0);
-}
-
-void
-animation_stop (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
-
- if (priv->playback_timer)
- {
- /* Stop playing by removing any playback timer. */
- g_source_remove (priv->playback_timer);
- priv->playback_timer = 0;
- g_signal_emit (animation, animation_signals[STOP], 0);
- }
-}
-
-void
-animation_next (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
- GeglBuffer *buffer = NULL;
- gint previous_pos = priv->position;
- gboolean identical;
-
- priv->position = animation_get_playback_start (animation) +
- ((priv->position - animation_get_playback_start (animation) + 1) %
- (animation_get_playback_stop (animation) - animation_get_playback_start (animation) +
1));
-
- identical = ANIMATION_GET_CLASS (animation)->same (animation,
- previous_pos,
- priv->position);
- if (! identical)
- {
- buffer = animation_get_frame (animation, priv->position);
- }
- g_signal_emit (animation, animation_signals[RENDER], 0,
- priv->position, buffer, ! identical);
- if (buffer != NULL)
- {
- g_object_unref (buffer);
- }
-}
-
-void
-animation_prev (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
- GeglBuffer *buffer = NULL;
- gint prev_pos = priv->position;
- gboolean identical;
-
- if (priv->position == animation_get_playback_start (animation))
- {
- priv->position = animation_get_playback_stop (animation);
- }
- else
- {
- --priv->position;
- }
-
- identical = ANIMATION_GET_CLASS (animation)->same (animation,
- prev_pos,
- priv->position);
- if (! identical)
- {
- buffer = animation_get_frame (animation, priv->position);
- }
- g_signal_emit (animation, animation_signals[RENDER], 0,
- priv->position, buffer, ! identical);
- if (buffer)
- g_object_unref (buffer);
-}
-
-void
-animation_jump (Animation *animation,
- gint index)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
- GeglBuffer *buffer = NULL;
- gint prev_pos = priv->position;
- gboolean identical;
-
- if (index < priv->playback_start ||
- index > priv->playback_stop)
- return;
- else
- priv->position = index;
-
- identical = ANIMATION_GET_CLASS (animation)->same (animation,
- prev_pos,
- priv->position);
- if (! identical)
- {
- buffer = animation_get_frame (animation, priv->position);
- }
- g_signal_emit (animation, animation_signals[RENDER], 0,
- priv->position, buffer, ! identical);
- if (buffer)
- g_object_unref (buffer);
-}
-
-void
-animation_set_playback_start (Animation *animation,
- gint frame_number)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
- gint duration;
-
- duration = animation_get_duration (animation);
-
- if (frame_number < 0 ||
- frame_number >= duration)
- {
- priv->playback_start = 0;
- }
- else
- {
- priv->playback_start = frame_number;
- }
- if (priv->playback_stop < priv->playback_start)
- {
- priv->playback_stop = duration - 1;
- }
-
- g_signal_emit (animation, animation_signals[PLAYBACK_RANGE], 0,
- priv->playback_start, priv->playback_stop,
- duration);
-
- if (priv->position < priv->playback_start ||
- priv->position > priv->playback_stop)
- {
- animation_jump (animation, priv->playback_start);
- }
-}
-
-gint
-animation_get_playback_start (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
-
- return priv->playback_start;
-}
-
-void
-animation_set_playback_stop (Animation *animation,
- gint frame_number)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
- gint duration;
-
- duration = animation_get_duration (animation);
-
- if (frame_number < 0 ||
- frame_number >= duration)
- {
- priv->playback_stop = duration - 1;
- }
- else
- {
- priv->playback_stop = frame_number;
- }
- if (priv->playback_stop < priv->playback_start)
- {
- priv->playback_start = 0;
- }
- g_signal_emit (animation, animation_signals[PLAYBACK_RANGE], 0,
- priv->playback_start, priv->playback_stop,
- duration);
-
- if (priv->position < priv->playback_start ||
- priv->position > priv->playback_stop)
- {
- animation_jump (animation, priv->playback_start);
- }
-}
-
-gint
-animation_get_playback_stop (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
-
- return priv->playback_stop;
-}
-
void
animation_set_framerate (Animation *animation,
gdouble fps)
@@ -818,6 +478,18 @@ animation_loaded (Animation *animation)
/************ Private Functions ****************/
static void
+animation_finalize (GObject *object)
+{
+ Animation *animation = ANIMATION (object);
+ AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
+
+ if (priv->xml)
+ g_free (priv->xml);
+
+ G_OBJECT_CLASS (parent_class)->finalize (object);
+}
+
+static void
animation_set_property (GObject *object,
guint property_id,
const GValue *value,
@@ -867,49 +539,3 @@ animation_real_same (Animation *animation,
/* By default all frames are supposed different. */
return (previous_pos == next_pos);
}
-
-static gboolean
-animation_advance_frame_callback (Animation *animation)
-{
- AnimationPrivate *priv = ANIMATION_GET_PRIVATE (animation);
- gint64 duration;
- gint64 duration_since_start;
- static gboolean prev_low_framerate = FALSE;
-
- animation_next (animation);
- duration = (gint) (1000.0 / animation_get_framerate (animation));
-
- duration_since_start = (g_get_monotonic_time () - priv->playback_start_time) / 1000;
- duration = duration - (duration_since_start - priv->frames_played * duration);
-
- if (duration < 1)
- {
- if (prev_low_framerate)
- {
- /* Let's only warn the user for several subsequent slow frames. */
- gdouble real_framerate = (gdouble) priv->frames_played * 1000.0 / duration_since_start;
- if (real_framerate < priv->framerate)
- g_signal_emit (animation, animation_signals[LOW_FRAMERATE], 0,
- real_framerate);
- }
- duration = 1;
- prev_low_framerate = TRUE;
- }
- else
- {
- if (prev_low_framerate)
- {
- /* Let's reset framerate warning. */
- g_signal_emit (animation, animation_signals[LOW_FRAMERATE], 0,
- animation_get_framerate (animation));
- }
- prev_low_framerate = FALSE;
- }
- priv->frames_played++;
-
- priv->playback_timer = g_timeout_add ((guint) duration,
- (GSourceFunc) animation_advance_frame_callback,
- (Animation *) animation);
-
- return G_SOURCE_REMOVE;
-}
diff --git a/plug-ins/animation-play/core/animation.h b/plug-ins/animation-play/core/animation.h
index cf826b5..de47fbc 100644
--- a/plug-ins/animation-play/core/animation.h
+++ b/plug-ins/animation-play/core/animation.h
@@ -40,22 +40,35 @@ struct _AnimationClass
{
GObjectClass parent_class;
+ /* Signals */
+ void (*loading) (Animation *animation,
+ gdouble ratio);
+ void (*loaded) (Animation *animation);
+
+ void (*cache_invalidated) (Animation *animation,
+ gint position);
+ void (*duration_changed) (Animation *animation,
+ gint duration);
+ void (*framerate_changed) (Animation *animation,
+ gdouble fps);
+ void (*proxy) (Animation *animation,
+ gdouble ratio);
+
/* Defaults to returning FALSE for any different position. */
gboolean (*same) (Animation *animation,
gint prev_pos,
gint next_pos);
/* These virtual methods must be implemented by any subclass. */
- void (*load) (Animation *animation);
- void (*load_xml) (Animation *animation,
- const gchar *xml);
- gint (*get_duration) (Animation *animation);
-
+ void (*load) (Animation *animation);
+ void (*load_xml) (Animation *animation,
+ const gchar *xml);
+ gint (*get_duration) (Animation *animation);
- GeglBuffer * (*get_frame) (Animation *animation,
- gint pos);
+ GeglBuffer * (*get_frame) (Animation *animation,
+ gint pos);
- gchar * (*serialize) (Animation *animation);
+ gchar * (*serialize) (Animation *animation);
};
GType animation_get_type (void);
@@ -70,7 +83,6 @@ void animation_load (Animation *animation);
void animation_save_to_parasite (Animation *animation);
-gint animation_get_position (Animation *animation);
gint animation_get_duration (Animation *animation);
void animation_set_proxy (Animation *animation,
@@ -83,22 +95,6 @@ void animation_get_size (Animation *animation,
GeglBuffer * animation_get_frame (Animation *animation,
gint frame_number);
-
-gboolean animation_is_playing (Animation *animation);
-void animation_play (Animation *animation);
-void animation_stop (Animation *animation);
-void animation_next (Animation *animation);
-void animation_prev (Animation *animation);
-void animation_jump (Animation *animation,
- gint index);
-
-void animation_set_playback_start (Animation *animation,
- gint frame_number);
-gint animation_get_playback_start (Animation *animation);
-void animation_set_playback_stop (Animation *animation,
- gint frame_number);
-gint animation_get_playback_stop (Animation *animation);
-
void animation_set_framerate (Animation *animation,
gdouble fps);
gdouble animation_get_framerate (Animation *animation);
diff --git a/plug-ins/animation-play/core/animationanimatic.c
b/plug-ins/animation-play/core/animationanimatic.c
index 756876a..e792648 100644
--- a/plug-ins/animation-play/core/animationanimatic.c
+++ b/plug-ins/animation-play/core/animationanimatic.c
@@ -79,7 +79,7 @@ struct _AnimationAnimaticPrivate
ANIMATION_TYPE_ANIMATIC, \
AnimationAnimaticPrivate)
-static void animation_animatic_finalize (GObject *object);
+static void animation_animatic_finalize (GObject *object);
/* Virtual methods */
@@ -99,11 +99,11 @@ static gboolean animation_animatic_same (Animation *animatio
/* XML parsing */
static void animation_animatic_start_element (GMarkupParseContext *context,
- const gchar *element_name,
- const gchar **attribute_names,
- const gchar **attribute_values,
- gpointer user_data,
- GError **error);
+ const gchar *element_name,
+ const gchar **attribute_names,
+ const gchar **attribute_values,
+ gpointer user_data,
+ GError **error);
static void animation_animatic_end_element (GMarkupParseContext *context,
const gchar *element_name,
gpointer user_data,
@@ -120,8 +120,6 @@ static void animation_animatic_text (GMarkupParseContext *context
static void animation_animatic_cache (AnimationAnimatic *animation,
gint panel,
gboolean recursion);
-static gint animation_animatic_get_layer (AnimationAnimatic *animation,
- gint pos);
/* Tag handling (from layer names) */
@@ -229,50 +227,19 @@ animation_animatic_set_panel_duration (AnimationAnimatic *animatic,
{
AnimationAnimaticPrivate *priv = GET_PRIVATE (animatic);
Animation *animation = ANIMATION (animatic);
- gint prev_length = animation_get_duration (animation);
- gint playback_start = animation_get_playback_start (animation);
- gint playback_stop = animation_get_playback_stop (animation);
- gint position = animation_get_position (animation);
- gint layer_id;
- gint length;
+ gint duration;
g_return_if_fail (panel_duration >= 0 &&
panel_num >= 0 &&
panel_num < priv->n_panels);
- layer_id = animation_animatic_get_layer (animatic, position);
-
priv->durations[panel_num] = panel_duration;
- length = animation_get_duration (animation);
+ duration = animation_get_duration (animation);
- if (playback_start >= length)
- {
- playback_start = 0;
- }
- if (playback_stop >= length ||
- playback_stop == prev_length - 1)
- {
- playback_stop = length - 1;
- }
g_signal_emit (animatic, signals[PANEL_DURATION], 0,
panel_num, panel_duration);
- g_signal_emit_by_name (animatic, "playback-range",
- playback_start, playback_stop,
- animation_get_duration (animation));
- if (position >= length)
- {
- animation_jump (animation, length - 1);
- }
- else if (layer_id != animation_animatic_get_layer (animatic, position))
- {
- GeglBuffer *buffer;
-
- buffer = animation_get_frame (animation, position);
- g_signal_emit_by_name (animation, "render",
- position, buffer, TRUE);
- if (buffer)
- g_object_unref (buffer);
- }
+ g_signal_emit_by_name (animation, "duration-changed",
+ duration);
}
gint
@@ -313,6 +280,7 @@ animation_animatic_get_comment (AnimationAnimatic *animatic,
g_return_val_if_fail (panel_num >= 0 &&
panel_num < priv->n_panels,
0);
+
return priv->comments[panel_num];
}
@@ -342,6 +310,7 @@ animation_animatic_get_combine (AnimationAnimatic *animatic,
g_return_val_if_fail (panel_num >= 0 &&
panel_num < priv->n_panels,
FALSE);
+
return priv->combine[panel_num];
}
@@ -370,22 +339,23 @@ animation_animatic_get_panel (AnimationAnimatic *animation,
return -1;
}
-void animation_animatic_jump_panel (AnimationAnimatic *animation,
- gint panel)
+gint
+animation_animatic_get_position (AnimationAnimatic *animation,
+ gint panel)
{
AnimationAnimaticPrivate *priv = GET_PRIVATE (animation);
- /* Get the first frame position for a given panel. */
gint pos = 0;
gint i;
- g_return_if_fail (panel < priv->n_panels);
+ g_return_val_if_fail (panel < priv->n_panels,
+ priv->n_panels - 1);
for (i = 0; i < panel; i++)
{
pos += priv->durations[i];
}
- animation_jump (ANIMATION (animation), pos);
+ return pos;
}
/**** Virtual methods ****/
@@ -868,7 +838,7 @@ animation_animatic_cache (AnimationAnimatic *animatic,
GeglBuffer *backdrop_buffer = NULL;
gint layer_offx;
gint layer_offy;
- gint position;
+ /*gint position;*/
gint preview_width;
gint preview_height;
gint32 image_id;
@@ -907,6 +877,11 @@ animation_animatic_cache (AnimationAnimatic *animatic,
layer_offx, layer_offy);
g_object_unref (buffer);
+ g_signal_emit_by_name (animation, "cache-invalidated",
+ animation_animatic_get_position (animatic,
+ panel),
+ 1);
+
/* If next panel is in "combine" mode, it must also be re-cached.
* And so on, recursively. */
if (recursion &&
@@ -915,42 +890,6 @@ animation_animatic_cache (AnimationAnimatic *animatic,
{
animation_animatic_cache (animatic, panel + 1, TRUE);
}
-
- /* Finally re-render if we are currently showing this panel. */
- position = animation_get_position (animation);
- if (animation_animatic_get_panel (animatic, position) == panel)
- {
- buffer = animation_get_frame (animation, position);
- g_signal_emit_by_name (animation, "render",
- position, buffer, TRUE);
- if (buffer)
- {
- g_object_unref (buffer);
- }
- }
-}
-
-static gint
-animation_animatic_get_layer (AnimationAnimatic *animation,
- gint pos)
-{
- AnimationAnimaticPrivate *priv = GET_PRIVATE (animation);
- gint count = 0;
- gint i = -1;
-
- if (priv->n_panels > 0 &&
- pos >= 0 &&
- pos < animation_animatic_get_duration (ANIMATION (animation)))
- {
- for (i = priv->n_panels - 1; i >= 0; i--)
- {
- count += priv->durations[i];
- if (count >= pos)
- break;
- }
- }
-
- return i;
}
/**** TAG UTILS ****/
diff --git a/plug-ins/animation-play/core/animationanimatic.h
b/plug-ins/animation-play/core/animationanimatic.h
index 12e6180..fccd85c 100644
--- a/plug-ins/animation-play/core/animationanimatic.h
+++ b/plug-ins/animation-play/core/animationanimatic.h
@@ -46,25 +46,26 @@ struct _AnimationAnimaticClass
GType animation_animatic_get_type (void);
void animation_animatic_set_panel_duration (AnimationAnimatic *animatic,
- gint panel_num,
+ gint panel,
gint duration);
gint animation_animatic_get_panel_duration (AnimationAnimatic *animatic,
- gint panel_num);
+ gint panel);
-void animation_animatic_set_comment (AnimationAnimatic *animatic,
- gint panel_num,
- const gchar *comment);
-const gchar * animation_animatic_get_comment (AnimationAnimatic *animatic,
- gint panel_num);
+void animation_animatic_set_comment (AnimationAnimatic *animatic,
+ gint panel,
+ const gchar *comment);
+const gchar * animation_animatic_get_comment (AnimationAnimatic *animatic,
+ gint panel);
-void animation_animatic_set_combine (AnimationAnimatic *animatic,
- gint panel_num,
- gboolean combine);
-const gboolean animation_animatic_get_combine (AnimationAnimatic *animatic,
- gint panel_num);
+void animation_animatic_set_combine (AnimationAnimatic *animatic,
+ gint panel,
+ gboolean combine);
+const gboolean animation_animatic_get_combine (AnimationAnimatic *animatic,
+ gint panel);
+
+gint animation_animatic_get_panel (AnimationAnimatic *animation,
+ gint pos);
+gint animation_animatic_get_position (AnimationAnimatic *animation,
+ gint panel);
-gint animation_animatic_get_panel (AnimationAnimatic *animation,
- gint pos);
-void animation_animatic_jump_panel (AnimationAnimatic *animation,
- gint panel);
#endif /* __ANIMATION_ANIMATIC_H__ */
diff --git a/plug-ins/animation-play/widgets/animation-dialog.c
b/plug-ins/animation-play/widgets/animation-dialog.c
index 184ba87..35d8f3d 100755
--- a/plug-ins/animation-play/widgets/animation-dialog.c
+++ b/plug-ins/animation-play/widgets/animation-dialog.c
@@ -31,6 +31,7 @@
#include "core/animation.h"
#include "core/animationanimatic.h"
#include "core/animation-celanimation.h"
+#include "core/animation-playback.h"
#include "animation-dialog.h"
#include "animation-layer-view.h"
@@ -57,54 +58,55 @@ typedef struct _AnimationDialogPrivate AnimationDialogPrivate;
struct _AnimationDialogPrivate
{
- gint32 image_id;
+ gint32 image_id;
- Animation *animation;
- gdouble zoom;
+ Animation *animation;
+ AnimationPlayback *playback;
+ gdouble zoom;
/* GUI */
- GtkWidget *play_bar;
- GtkWidget *progress_bar;
- GtkWidget *settings_bar;
+ GtkWidget *play_bar;
+ GtkWidget *progress_bar;
+ GtkWidget *settings_bar;
- GtkWidget *animation_type_combo;
- GtkWidget *fpscombo;
- GtkWidget *zoomcombo;
- GtkWidget *proxycombo;
+ GtkWidget *animation_type_combo;
+ GtkWidget *fpscombo;
+ GtkWidget *zoomcombo;
+ GtkWidget *proxycombo;
- GtkWidget *progress;
- GtkWidget *startframe_spin;
- GtkWidget *endframe_spin;
+ GtkWidget *progress;
+ GtkWidget *startframe_spin;
+ GtkWidget *endframe_spin;
- GtkWidget *view_bar;
- GtkWidget *refresh;
+ GtkWidget *view_bar;
+ GtkWidget *refresh;
- GtkWidget *drawing_area;
- guchar *drawing_area_data;
- guint drawing_area_width;
- guint drawing_area_height;
+ GtkWidget *drawing_area;
+ guchar *drawing_area_data;
+ guint drawing_area_width;
+ guint drawing_area_height;
- GtkWidget *shape_window;
- GtkWidget *shape_drawing_area;
- guchar *shape_drawing_area_data;
- guint shape_drawing_area_width;
- guint shape_drawing_area_height;
+ GtkWidget *shape_window;
+ GtkWidget *shape_drawing_area;
+ guchar *shape_drawing_area_data;
+ guint shape_drawing_area_width;
+ guint shape_drawing_area_height;
/* The hpaned (left is preview, right is layer list. */
- GtkWidget *hpaned;
- GtkWidget *layer_list;
+ GtkWidget *hpaned;
+ GtkWidget *layer_list;
/* The vpaned (bottom is timeline, above is preview). */
- GtkWidget *vpaned;
- GtkWidget *xsheet;
+ GtkWidget *vpaned;
+ GtkWidget *xsheet;
/* Actions */
- GtkUIManager *ui_manager;
+ GtkUIManager *ui_manager;
- GtkActionGroup *play_actions;
- GtkActionGroup *settings_actions;
- GtkActionGroup *view_actions;
- GtkActionGroup *various_actions;
+ GtkActionGroup *play_actions;
+ GtkActionGroup *settings_actions;
+ GtkActionGroup *view_actions;
+ GtkActionGroup *various_actions;
};
#define GET_PRIVATE(dialog) \
@@ -198,24 +200,14 @@ static gboolean popup_menu (GtkWidget *widget,
static void show_loading_progress (Animation *animation,
gdouble load_rate,
AnimationDialog *dialog);
-static void animation_update_progress (Animation *animation,
- gint num_frames,
- gint playback_start,
- gint playback_stop,
- AnimationDialog *dialog);
-static void block_ui (Animation *animation,
- AnimationDialog *dialog);
static void unblock_ui (Animation *animation,
- gint num_frames,
- gint playback_start,
- gint playback_stop,
- gint preview_width,
- gint preview_height,
AnimationDialog *dialog);
-static void playback_range_changed (Animation *animation,
+static void duration_changed (AnimationPlayback *playback,
+ gint duration,
+ AnimationDialog *dialog);
+static void playback_range_changed (AnimationPlayback *playback,
gint playback_start,
gint playback_stop,
- gint length,
AnimationDialog *dialog);
static void proxy_changed (Animation *animation,
gdouble fps,
@@ -223,7 +215,7 @@ static void proxy_changed (Animation *animation,
static void framerate_changed (Animation *animation,
gdouble fps,
AnimationDialog *dialog);
-static void low_framerate_cb (Animation *animation,
+static void low_framerate_cb (AnimationPlayback *playback,
gdouble real_framerate,
AnimationDialog *dialog);
@@ -244,7 +236,7 @@ static gboolean shape_released (GtkWidget *widget);
static gboolean shape_motion (GtkWidget *widget,
GdkEventMotion *event);
-static void render_callback (Animation *animation,
+static void render_callback (AnimationPlayback *animation,
gint frame_number,
GeglBuffer *buffer,
gboolean must_draw_null,
@@ -276,6 +268,8 @@ static void show_goto_progress (AnimationDialog *dialog,
static void show_playing_progress (AnimationDialog *dialog);
/* Utils */
+static void update_progress (AnimationDialog *dialog);
+static void block_ui (AnimationDialog *dialog);
static gboolean is_detached (AnimationDialog *dialog);
static void play_pause (AnimationDialog *dialog);
@@ -313,6 +307,9 @@ animation_dialog_class_init (AnimationDialogClass *klass)
static void
animation_dialog_init (AnimationDialog *dialog)
{
+ AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
+
+ priv->playback = animation_playback_new ();
}
/**** Public Functions ****/
@@ -326,7 +323,6 @@ animation_dialog_new (gint32 image_id)
const gchar *xml = NULL;
gboolean animatic_selected = TRUE;
- /* If this animation has specific settings already, override the global ones. */
parasite = gimp_image_get_parasite (image_id,
PLUG_IN_PROC "/selected");
if (parasite)
@@ -367,7 +363,6 @@ animation_dialog_new (gint32 image_id)
NULL);
animation_dialog_set_animation (ANIMATION_DIALOG (dialog),
animation);
- animation_load (animation);
return dialog;
}
@@ -861,6 +856,9 @@ animation_dialog_finalize (GObject *object)
if (priv->animation)
g_object_unref (priv->animation);
+ if (priv->playback)
+ g_object_unref (priv->playback);
+
G_OBJECT_CLASS (parent_class)->finalize (object);
/* When this window ends, the plugin ends. */
@@ -1098,6 +1096,9 @@ animation_dialog_set_animation (AnimationDialog *dialog,
G_CALLBACK (framerate_changed),
dialog);
g_signal_handlers_disconnect_by_func (priv->animation,
+ G_CALLBACK (duration_changed),
+ dialog);
+ g_signal_handlers_disconnect_by_func (priv->playback,
G_CALLBACK (playback_range_changed),
dialog);
@@ -1105,15 +1106,12 @@ animation_dialog_set_animation (AnimationDialog *dialog,
(GCallback) show_loading_progress,
dialog);
g_signal_handlers_disconnect_by_func (priv->animation,
- (GCallback) block_ui,
- dialog);
- g_signal_handlers_disconnect_by_func (priv->animation,
(GCallback) unblock_ui,
dialog);
g_signal_handlers_disconnect_by_func (priv->animation,
G_CALLBACK (render_callback),
dialog);
- g_signal_handlers_disconnect_by_func (priv->animation,
+ g_signal_handlers_disconnect_by_func (priv->playback,
G_CALLBACK (low_framerate_cb),
dialog);
}
@@ -1168,7 +1166,7 @@ animation_dialog_set_animation (AnimationDialog *dialog,
G_CALLBACK (proxycombo_changed),
dialog);
/* Settings: fps */
- fps = animation_get_framerate (animation);
+ fps = animation_get_framerate (priv->animation);
gtk_combo_box_set_active (GTK_COMBO_BOX (priv->fpscombo), -1);
for (index = 0; index < 5; index++)
@@ -1242,7 +1240,8 @@ animation_dialog_set_animation (AnimationDialog *dialog,
gtk_frame_set_label (GTK_FRAME (frame), _("Storyboard"));
/* The Storyboard view. */
- storyboard = animation_storyboard_new (ANIMATION_ANIMATIC (animation));
+ storyboard = animation_storyboard_new (ANIMATION_ANIMATIC (animation),
+ priv->playback);
gtk_scrolled_window_add_with_viewport (GTK_SCROLLED_WINDOW (scrolled_win),
storyboard);
gtk_widget_show (storyboard);
@@ -1281,7 +1280,7 @@ animation_dialog_set_animation (AnimationDialog *dialog,
gtk_widget_show (frame);
priv->xsheet = animation_xsheet_new (ANIMATION_CEL_ANIMATION (animation),
- priv->layer_list);
+ priv->playback, priv->layer_list);
gtk_container_add (GTK_CONTAINER (frame), priv->xsheet);
gtk_widget_show (priv->xsheet);
}
@@ -1292,33 +1291,34 @@ animation_dialog_set_animation (AnimationDialog *dialog,
dialog);
/* Animation. */
- g_signal_connect (priv->animation, "proxy",
+ g_signal_connect (priv->animation, "proxy-changed",
G_CALLBACK (proxy_changed),
dialog);
g_signal_connect (priv->animation, "framerate-changed",
G_CALLBACK (framerate_changed),
dialog);
- g_signal_connect (priv->animation,
- "playback-range",
+ g_signal_connect (priv->animation, "duration-changed",
+ G_CALLBACK (duration_changed),
+ dialog);
+ g_signal_connect (priv->playback, "range",
G_CALLBACK (playback_range_changed),
dialog);
g_signal_connect (priv->animation, "loading",
(GCallback) show_loading_progress,
dialog);
- g_signal_connect (priv->animation, "loading-start",
- (GCallback) block_ui,
- dialog);
g_signal_connect (priv->animation, "loaded",
(GCallback) unblock_ui,
dialog);
- g_signal_connect (priv->animation, "render",
+ g_signal_connect (priv->playback, "render",
G_CALLBACK (render_callback),
dialog);
- g_signal_connect (priv->animation, "low-framerate-playback",
+ g_signal_connect (priv->playback, "low-framerate",
G_CALLBACK (low_framerate_cb),
dialog);
+ /* Set the playback */
+ animation_playback_set_animation (priv->playback, animation);
}
static void
@@ -1369,7 +1369,7 @@ update_ui_sensitivity (AnimationDialog *dialog)
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
gboolean animated;
- animated = animation_get_playback_stop (priv->animation) - animation_get_playback_start (priv->animation)
1;
+ animated = animation_playback_get_stop (priv->playback) - animation_playback_get_start (priv->playback) >
1;
/* Play actions only if we selected several frames between start/end. */
gtk_action_group_set_sensitive (priv->play_actions, animated);
gtk_widget_set_sensitive (GTK_WIDGET (priv->play_bar), animated);
@@ -1397,7 +1397,7 @@ close_callback (GtkAction *action,
{
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
- animation_stop (priv->animation);
+ animation_playback_stop (priv->playback);
gtk_widget_destroy (GTK_WIDGET (dialog));
}
@@ -1621,7 +1621,7 @@ proxycombo_activated (GtkEntry *combo_entry,
{
gboolean was_playing;
- was_playing = animation_is_playing (priv->animation);
+ was_playing = animation_playback_is_playing (priv->playback);
animation_set_proxy (priv->animation, ratio);
update_scale (dialog, get_zoom (dialog, -1));
@@ -1733,7 +1733,7 @@ adjustment_pressed (GtkWidget *widget,
GtkAdjustment *adj = gtk_spin_button_get_adjustment (spin);
gtk_adjustment_set_value (adj,
- (gdouble) animation_get_position (priv->animation) + 1.0);
+ (gdouble) animation_playback_get_position (priv->playback) + 1.0);
/* We don't want the middle click to have another usage (in
* particular, there is likely no need to copy-paste in these spin
@@ -1754,7 +1754,7 @@ startframe_changed (GtkAdjustment *adjustment,
if (! priv->animation)
return;
- animation_set_playback_start (priv->animation, (gint) value - 1);
+ animation_playback_set_start (priv->playback, (gint) value - 1);
update_ui_sensitivity (dialog);
}
@@ -1769,7 +1769,7 @@ endframe_changed (GtkAdjustment *adjustment,
if (! priv->animation)
return;
- animation_set_playback_stop (priv->animation, (gint) value - 1);
+ animation_playback_set_stop (priv->playback, (gint) value - 1);
update_ui_sensitivity (dialog);
}
@@ -1780,15 +1780,15 @@ play_callback (GtkToggleAction *action,
{
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
- if (! animation_is_playing (priv->animation))
+ if (! animation_playback_is_playing (priv->playback))
{
gtk_action_set_icon_name (GTK_ACTION (action), "media-playback-pause");
- animation_play (priv->animation);
+ animation_playback_play (priv->playback);
}
else
{
gtk_action_set_icon_name (GTK_ACTION (action), "media-playback-start");
- animation_stop (priv->animation);
+ animation_playback_stop (priv->playback);
/* The framerate combo might have been modified to display slowness
* warnings. */
@@ -1799,7 +1799,7 @@ play_callback (GtkToggleAction *action,
g_object_set (action,
"tooltip",
- animation_is_playing (priv->animation) ? _("Pause playback") : _("Start playback"),
+ animation_playback_is_playing (priv->playback) ? _("Pause playback") : _("Start playback"),
NULL);
}
@@ -1809,9 +1809,9 @@ step_back_callback (GtkAction *action,
{
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
- if (animation_is_playing (priv->animation))
+ if (animation_playback_is_playing (priv->playback))
play_pause (dialog);
- animation_prev (priv->animation);
+ animation_playback_prev (priv->playback);
}
static void
@@ -1820,9 +1820,9 @@ step_callback (GtkAction *action,
{
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
- if (animation_is_playing (priv->animation))
- animation_stop (priv->animation);
- animation_next (priv->animation);
+ if (animation_playback_is_playing (priv->playback))
+ animation_playback_stop (priv->playback);
+ animation_playback_next (priv->playback);
}
static void
@@ -1832,12 +1832,12 @@ rewind_callback (GtkAction *action,
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
gboolean was_playing;
- was_playing = animation_is_playing (priv->animation);
+ was_playing = animation_playback_is_playing (priv->playback);
if (was_playing)
play_pause (dialog);
- animation_jump (priv->animation, animation_get_playback_start (priv->animation));
+ animation_playback_jump (priv->playback, animation_playback_get_start (priv->playback));
/* If we were playing, start playing again. */
if (was_playing)
@@ -1899,7 +1899,7 @@ detach_callback (GtkToggleAction *action,
/* Force a refresh after detachment/attachment. */
buffer = animation_get_frame (priv->animation,
- animation_get_position (priv->animation));
+ animation_playback_get_position (priv->playback));
render_frame (dialog, buffer, TRUE);
/* clean up */
if (buffer != NULL)
@@ -1931,7 +1931,7 @@ show_loading_progress (Animation *animation,
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
gchar *text;
- block_ui (animation, dialog);
+ block_ui (dialog);
/* update the dialog's progress bar */
gtk_progress_bar_set_fraction (GTK_PROGRESS_BAR (priv->progress), load_rate);
@@ -1946,85 +1946,32 @@ show_loading_progress (Animation *animation,
}
static void
-animation_update_progress (Animation *animation,
- gint num_frames,
- gint playback_start,
- gint playback_stop,
- AnimationDialog *dialog)
-{
- AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
- gint frame_spin_size;
- gint last_frame = num_frames;
-
- frame_spin_size = (gint) (log10 (last_frame + 1 - (last_frame % 10))) + 1;
- gtk_entry_set_width_chars (GTK_ENTRY (priv->startframe_spin), frame_spin_size);
- gtk_entry_set_width_chars (GTK_ENTRY (priv->endframe_spin), frame_spin_size);
-
- gtk_adjustment_configure (gtk_spin_button_get_adjustment (GTK_SPIN_BUTTON (priv->startframe_spin)),
- (gdouble) playback_start + 1.0,
- 1.0,
- (gdouble) last_frame,
- 1.0, 5.0, 0.0);
- gtk_adjustment_configure (gtk_spin_button_get_adjustment (GTK_SPIN_BUTTON (priv->endframe_spin)),
- (gdouble) playback_stop + 1.0,
- (gdouble) playback_start + 1.0,
- (gdouble) last_frame,
- 1.0, 5.0, 0.0);
-
- update_ui_sensitivity (dialog);
-}
-
-static void
-block_ui (Animation *animation,
- AnimationDialog *dialog)
+unblock_ui (Animation *animation,
+ AnimationDialog *dialog)
{
- AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
-
- if (animation_is_playing (priv->animation))
- play_pause (dialog);
-
- gtk_action_group_set_sensitive (priv->play_actions, FALSE);
- gtk_widget_set_sensitive (priv->play_bar, FALSE);
- gtk_widget_set_sensitive (priv->progress_bar, FALSE);
- gtk_action_group_set_sensitive (priv->settings_actions, FALSE);
- gtk_widget_set_sensitive (GTK_WIDGET (priv->settings_bar), FALSE);
- gtk_action_group_set_sensitive (priv->view_actions, FALSE);
- gtk_widget_set_sensitive (GTK_WIDGET (priv->view_bar), FALSE);
+ update_progress (dialog);
+ animation_dialog_refresh (dialog);
}
static void
-unblock_ui (Animation *animation,
- gint num_frames,
- gint playback_start,
- gint playback_stop,
- gint preview_width,
- gint preview_height,
- AnimationDialog *dialog)
+duration_changed (AnimationPlayback *playback,
+ gint duration,
+ AnimationDialog *dialog)
{
- animation_update_progress (animation,
- num_frames,
- playback_start,
- playback_stop,
- dialog);
- animation_dialog_refresh (dialog);
+ update_progress (dialog);
}
static void
-playback_range_changed (Animation *animation,
- gint playback_start,
- gint playback_stop,
- gint length,
- AnimationDialog *dialog)
+playback_range_changed (AnimationPlayback *playback,
+ gint playback_start,
+ gint playback_stop,
+ AnimationDialog *dialog)
{
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
GtkAdjustment *startframe_adjust;
GtkAdjustment *stopframe_adjust;
- animation_update_progress (animation,
- length,
- playback_start,
- playback_stop,
- dialog);
+ update_progress (dialog);
startframe_adjust = gtk_spin_button_get_adjustment (GTK_SPIN_BUTTON (priv->startframe_spin));
stopframe_adjust = gtk_spin_button_get_adjustment (GTK_SPIN_BUTTON (priv->endframe_spin));
@@ -2100,9 +2047,9 @@ framerate_changed (Animation *animation,
}
static void
-low_framerate_cb (Animation *animation,
- gdouble real_framerate,
- AnimationDialog *dialog)
+low_framerate_cb (AnimationPlayback *playback,
+ gdouble real_framerate,
+ AnimationDialog *dialog)
{
AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
gchar *text;
@@ -2290,7 +2237,7 @@ da_size_callback (GtkWidget *drawing_area,
GeglBuffer *buffer;
buffer = animation_get_frame (priv->animation,
- animation_get_position (priv->animation));
+ animation_playback_get_position (priv->playback));
render_frame (dialog, buffer, TRUE);
/* clean up */
if (buffer != NULL)
@@ -2371,11 +2318,11 @@ shape_motion (GtkWidget *widget,
}
static void
-render_callback (Animation *animation,
- gint frame_number,
- GeglBuffer *buffer,
- gboolean must_draw_null,
- AnimationDialog *dialog)
+render_callback (AnimationPlayback *playback,
+ gint frame_number,
+ GeglBuffer *buffer,
+ gboolean must_draw_null,
+ AnimationDialog *dialog)
{
render_frame (dialog, buffer, must_draw_null);
@@ -2391,7 +2338,7 @@ render_on_realize (GtkWidget *drawing_area,
GeglBuffer *buffer;
buffer = animation_get_frame (priv->animation,
- animation_get_position (priv->animation));
+ animation_playback_get_position (priv->playback));
render_frame (dialog, buffer, TRUE);
/* clean up */
if (buffer != NULL)
@@ -2590,7 +2537,7 @@ progress_button (GtkWidget *widget,
((gdouble) allocation.width /
((gdouble) duration - 0.99)));
- animation_jump (priv->animation, frame);
+ animation_playback_jump (priv->playback, frame);
}
return FALSE;
@@ -2678,17 +2625,64 @@ show_playing_progress (AnimationDialog *dialog)
/* update the dialog's progress bar */
gtk_progress_bar_set_fraction (GTK_PROGRESS_BAR (priv->progress),
- ((gfloat) animation_get_position (priv->animation) /
+ ((gfloat) animation_playback_get_position (priv->playback) /
(gfloat) (animation_get_duration (priv->animation) - 0.999)));
text = g_strdup_printf (_("Frame %d of %d"),
- animation_get_position (priv->animation) + 1,
+ animation_playback_get_position (priv->playback) + 1,
animation_get_duration (priv->animation));
gtk_progress_bar_set_text (GTK_PROGRESS_BAR (priv->progress), text);
g_free (text);
}
+static void
+update_progress (AnimationDialog *dialog)
+{
+ AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
+ gint frame_spin_size;
+ gint duration = animation_get_duration (priv->animation);
+ gint playback_start;
+ gint playback_stop;
+
+ playback_start = animation_playback_get_start (priv->playback);
+ playback_stop = animation_playback_get_stop (priv->playback);
+
+ frame_spin_size = (gint) (log10 (duration + 1 - (duration % 10))) + 1;
+ gtk_entry_set_width_chars (GTK_ENTRY (priv->startframe_spin), frame_spin_size);
+ gtk_entry_set_width_chars (GTK_ENTRY (priv->endframe_spin), frame_spin_size);
+
+ gtk_adjustment_configure (gtk_spin_button_get_adjustment (GTK_SPIN_BUTTON (priv->startframe_spin)),
+ (gdouble) playback_start + 1.0,
+ 1.0,
+ (gdouble) duration,
+ 1.0, 5.0, 0.0);
+ gtk_adjustment_configure (gtk_spin_button_get_adjustment (GTK_SPIN_BUTTON (priv->endframe_spin)),
+ (gdouble) playback_stop + 1.0,
+ (gdouble) playback_start + 1.0,
+ (gdouble) duration,
+ 1.0, 5.0, 0.0);
+
+ update_ui_sensitivity (dialog);
+}
+
+static void
+block_ui (AnimationDialog *dialog)
+{
+ AnimationDialogPrivate *priv = GET_PRIVATE (dialog);
+
+ if (animation_playback_is_playing (priv->playback))
+ play_pause (dialog);
+
+ gtk_action_group_set_sensitive (priv->play_actions, FALSE);
+ gtk_widget_set_sensitive (priv->play_bar, FALSE);
+ gtk_widget_set_sensitive (priv->progress_bar, FALSE);
+ gtk_action_group_set_sensitive (priv->settings_actions, FALSE);
+ gtk_widget_set_sensitive (GTK_WIDGET (priv->settings_bar), FALSE);
+ gtk_action_group_set_sensitive (priv->view_actions, FALSE);
+ gtk_widget_set_sensitive (GTK_WIDGET (priv->view_bar), FALSE);
+}
+
static gboolean
is_detached (AnimationDialog *dialog)
{
diff --git a/plug-ins/animation-play/widgets/animation-storyboard.c
b/plug-ins/animation-play/widgets/animation-storyboard.c
index 613c2de..feaaa30 100644
--- a/plug-ins/animation-play/widgets/animation-storyboard.c
+++ b/plug-ins/animation-play/widgets/animation-storyboard.c
@@ -28,6 +28,7 @@
#include "libgimp/stdplugins-intl.h"
#include "core/animationanimatic.h"
+#include "core/animation-playback.h"
#include "animation-storyboard.h"
@@ -41,6 +42,7 @@ enum
struct _AnimationStoryboardPrivate
{
AnimationAnimatic *animation;
+ AnimationPlayback *playback;
gint current_panel;
@@ -63,20 +65,15 @@ static void animation_storyboard_finalize (GObject *obj
/* Callbacks on animation */
static void animation_storyboard_load (Animation *animation,
- G_GNUC_UNUSED gint num_frames,
- G_GNUC_UNUSED gint playback_start,
- G_GNUC_UNUSED gint playback_stop,
- G_GNUC_UNUSED gint preview_width,
- G_GNUC_UNUSED gint preview_height,
AnimationStoryboard *view);
-static void animation_storyboard_rendered (Animation *animation,
+static void animation_storyboard_rendered (AnimationPlayback *playback,
gint frame_number,
GeglBuffer *buffer,
gboolean must_draw_null,
AnimationStoryboard *view);
static void animation_storyboard_duration_spin_changed (GtkSpinButton *spinbutton,
- AnimationAnimatic *animation);
+ AnimationStoryboard *storyboard);
static gboolean animation_storyboard_comment_keypress (GtkWidget *entry,
GdkEventKey *event,
@@ -86,7 +83,7 @@ static void animation_storyboard_comment_changed (GtkTextBuffer *tex
static void animation_storyboard_disposal_toggled (GtkToggleButton *button,
AnimationAnimatic *animatic);
static void animation_storyboard_button_clicked (GtkWidget *widget,
- AnimationAnimatic *animatic);
+ AnimationStoryboard *storyboard);
G_DEFINE_TYPE (AnimationStoryboard, animation_storyboard, GTK_TYPE_TABLE)
@@ -136,13 +133,17 @@ animation_storyboard_init (AnimationStoryboard *view)
* Creates a new layer view tied to @animation, ready to be displayed.
*/
GtkWidget *
-animation_storyboard_new (AnimationAnimatic *animation)
+animation_storyboard_new (AnimationAnimatic *animation,
+ AnimationPlayback *playback)
{
- GtkWidget *layer_view;
+ GtkWidget *layer_view;
+ AnimationStoryboard *storyboard;
layer_view = g_object_new (ANIMATION_TYPE_STORYBOARD,
"animation", animation,
NULL);
+ storyboard = ANIMATION_STORYBOARD (layer_view);
+ storyboard->priv->playback = playback;
return layer_view;
}
@@ -228,11 +229,6 @@ animation_storyboard_finalize (GObject *object)
*/
static void
animation_storyboard_load (Animation *animation,
- gint num_frames,
- gint playback_start,
- gint playback_stop,
- gint preview_width,
- gint preview_height,
AnimationStoryboard *view)
{
AnimationAnimatic *animatic = ANIMATION_ANIMATIC (animation);
@@ -299,7 +295,7 @@ animation_storyboard_load (Animation *animation,
GINT_TO_POINTER (panel_num));
g_signal_connect (panel_button, "clicked",
G_CALLBACK (animation_storyboard_button_clicked),
- animation);
+ view);
view->priv->panel_buttons = g_list_prepend (view->priv->panel_buttons,
panel_button);
@@ -392,7 +388,7 @@ animation_storyboard_load (Animation *animation,
GINT_TO_POINTER (panel_num));
g_signal_connect (duration, "value-changed",
(GCallback) animation_storyboard_duration_spin_changed,
- animation);
+ view);
gtk_widget_show (duration);
disposal = gtk_toggle_button_new ();
@@ -419,14 +415,14 @@ animation_storyboard_load (Animation *animation,
disposal);
gtk_widget_show (disposal);
}
- g_signal_connect (animation, "render",
+ g_signal_connect (view->priv->playback, "render",
(GCallback) animation_storyboard_rendered,
view);
gimp_image_delete (image_id);
}
static void
-animation_storyboard_rendered (Animation *animation,
+animation_storyboard_rendered (AnimationPlayback *playback,
gint frame_number,
GeglBuffer *buffer,
gboolean must_draw_null,
@@ -436,7 +432,7 @@ animation_storyboard_rendered (Animation *animation,
GtkWidget *arrow;
gint panel;
- panel = animation_animatic_get_panel (ANIMATION_ANIMATIC (animation),
+ panel = animation_animatic_get_panel (view->priv->animation,
frame_number);
if (view->priv->current_panel >= 0)
{
@@ -456,18 +452,33 @@ animation_storyboard_rendered (Animation *animation,
}
static void
-animation_storyboard_duration_spin_changed (GtkSpinButton *spinbutton,
- AnimationAnimatic *animation)
+animation_storyboard_duration_spin_changed (GtkSpinButton *spinbutton,
+ AnimationStoryboard *storyboard)
{
gpointer panel_num;
gint duration;
+ gint panel_position;
+ gint position;
panel_num = g_object_get_data (G_OBJECT (spinbutton), "panel-num");
duration = gtk_spin_button_get_value_as_int (spinbutton);
- animation_animatic_set_panel_duration (animation,
+ position = animation_playback_get_position (storyboard->priv->playback);
+ panel_position = animation_animatic_get_position (storyboard->priv->animation,
+ GPOINTER_TO_INT (panel_num));
+ if (position >= panel_position)
+ {
+ gint cur_duration;
+
+ cur_duration = animation_animatic_get_panel_duration (storyboard->priv->animation,
+ GPOINTER_TO_INT (panel_num));
+ position += duration - cur_duration;
+ }
+
+ animation_animatic_set_panel_duration (storyboard->priv->animation,
GPOINTER_TO_INT (panel_num),
duration);
+ animation_playback_jump (storyboard->priv->playback, position);
}
static gboolean
@@ -535,11 +546,14 @@ animation_storyboard_disposal_toggled (GtkToggleButton *button,
}
static void
-animation_storyboard_button_clicked (GtkWidget *widget,
- AnimationAnimatic *animatic)
+animation_storyboard_button_clicked (GtkWidget *widget,
+ AnimationStoryboard *storyboard)
{
gpointer panel_num;
+ gint position;
panel_num = g_object_get_data (G_OBJECT (widget), "panel-num");
- animation_animatic_jump_panel (animatic, GPOINTER_TO_INT (panel_num));
+ position = animation_animatic_get_position (storyboard->priv->animation,
+ GPOINTER_TO_INT (panel_num));
+ animation_playback_jump (storyboard->priv->playback, position);
}
diff --git a/plug-ins/animation-play/widgets/animation-storyboard.h
b/plug-ins/animation-play/widgets/animation-storyboard.h
index f92a23e..cb7452b 100644
--- a/plug-ins/animation-play/widgets/animation-storyboard.h
+++ b/plug-ins/animation-play/widgets/animation-storyboard.h
@@ -46,7 +46,8 @@ struct _AnimationStoryboardClass
GType animation_storyboard_get_type (void) G_GNUC_CONST;
-GtkWidget * animation_storyboard_new (AnimationAnimatic *animation);
+GtkWidget * animation_storyboard_new (AnimationAnimatic *animation,
+ AnimationPlayback *playback);
#endif /* __ANIMATION_STORYBOARD_H__ */
diff --git a/plug-ins/animation-play/widgets/animation-xsheet.c
b/plug-ins/animation-play/widgets/animation-xsheet.c
index 4161f3d..c7aaa9a 100755
--- a/plug-ins/animation-play/widgets/animation-xsheet.c
+++ b/plug-ins/animation-play/widgets/animation-xsheet.c
@@ -25,6 +25,8 @@
#include <libgimp/gimp.h>
+#include "core/animation.h"
+#include "core/animation-playback.h"
#include "core/animation-celanimation.h"
#include "animation-layer-view.h"
@@ -42,6 +44,8 @@ enum
struct _AnimationXSheetPrivate
{
AnimationCelAnimation *animation;
+ AnimationPlayback *playback;
+
GtkWidget *layer_view;
GtkWidget *track_layout;
@@ -72,17 +76,13 @@ static void animation_xsheet_reset_layout (AnimationXSheet *xsheet);
/* Callbacks on animation. */
static void on_animation_loaded (Animation *animation,
- gint num_frames,
- gint playback_start,
- gint playback_stop,
- gint preview_width,
- gint preview_height,
- AnimationXSheet *xsheet);
-static void on_animation_rendered (Animation *animation,
- gint frame_number,
- GeglBuffer *buffer,
- gboolean must_draw_null,
AnimationXSheet *xsheet);
+/* Callbacks on callback. */
+static void on_animation_rendered (AnimationPlayback *animation,
+ gint frame_number,
+ GeglBuffer *buffer,
+ gboolean must_draw_null,
+ AnimationXSheet *xsheet);
/* Callbacks on layer view. */
static void on_layer_selection (AnimationLayerView *view,
@@ -160,6 +160,7 @@ animation_xsheet_init (AnimationXSheet *xsheet)
GtkWidget *
animation_xsheet_new (AnimationCelAnimation *animation,
+ AnimationPlayback *playback,
GtkWidget *layer_view)
{
GtkWidget *xsheet;
@@ -168,6 +169,10 @@ animation_xsheet_new (AnimationCelAnimation *animation,
"animation", animation,
"layer-view", layer_view,
NULL);
+ ANIMATION_XSHEET (xsheet)->priv->playback = playback;
+ g_signal_connect (ANIMATION_XSHEET (xsheet)->priv->playback,
+ "render",
+ G_CALLBACK (on_animation_rendered), xsheet);
return xsheet;
}
@@ -199,8 +204,6 @@ animation_xsheet_constructed (GObject *object)
/* Reload everything when we reload the animation. */
g_signal_connect_after (xsheet->priv->animation, "loaded",
G_CALLBACK (on_animation_loaded), xsheet);
- g_signal_connect (xsheet->priv->animation, "render",
- G_CALLBACK (on_animation_rendered), xsheet);
}
static void
@@ -254,6 +257,9 @@ animation_xsheet_finalize (GObject *object)
{
AnimationXSheet *xsheet = ANIMATION_XSHEET (object);
+ g_signal_handlers_disconnect_by_func (ANIMATION_XSHEET (xsheet)->priv->playback,
+ G_CALLBACK (on_animation_rendered),
+ xsheet);
if (xsheet->priv->animation)
g_object_unref (xsheet->priv->animation);
if (xsheet->priv->layer_view)
@@ -478,22 +484,17 @@ on_layer_selection (AnimationLayerView *view,
static void
on_animation_loaded (Animation *animation,
- gint num_frames,
- gint playback_start,
- gint playback_stop,
- gint preview_width,
- gint preview_height,
AnimationXSheet *xsheet)
{
animation_xsheet_reset_layout (xsheet);
}
static void
-on_animation_rendered (Animation *animation,
- gint frame_number,
- GeglBuffer *buffer,
- gboolean must_draw_null,
- AnimationXSheet *xsheet)
+on_animation_rendered (AnimationPlayback *playback,
+ gint frame_number,
+ GeglBuffer *buffer,
+ gboolean must_draw_null,
+ AnimationXSheet *xsheet)
{
GtkWidget *button;
@@ -522,8 +523,8 @@ animation_xsheet_frame_clicked (GtkWidget *button,
position = g_object_get_data (G_OBJECT (button), "frame-position");
- animation_jump (ANIMATION (xsheet->priv->animation),
- GPOINTER_TO_INT (position) + 1);
+ animation_playback_jump (xsheet->priv->playback,
+ GPOINTER_TO_INT (position));
if (xsheet->priv->active_pos_button)
{
GtkToggleButton *active_button;
diff --git a/plug-ins/animation-play/widgets/animation-xsheet.h
b/plug-ins/animation-play/widgets/animation-xsheet.h
index 9346419..4a4dfba 100755
--- a/plug-ins/animation-play/widgets/animation-xsheet.h
+++ b/plug-ins/animation-play/widgets/animation-xsheet.h
@@ -47,6 +47,7 @@ struct _AnimationXSheetClass
GType animation_xsheet_get_type (void) G_GNUC_CONST;
GtkWidget * animation_xsheet_new (AnimationCelAnimation *animation,
+ AnimationPlayback *playback,
GtkWidget *layer_view);
#endif /* __ANIMATION_XSHEET_H__ */
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]