1 /* ev-transition-animation.c
2 * this file is part of evince, a gnome document viewer
4 * Copyright (C) 2007 Carlos Garnacho <carlos@imendio.com>
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Library General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Library General Public License for more details.
16 * You should have received a copy of the GNU Library General Public
17 * License along with this library; if not, write to the
18 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19 * Boston, MA 02111-1307, USA.
24 #include "ev-transition-animation.h"
25 #include "ev-timeline.h"
27 #define EV_TRANSITION_ANIMATION_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE ((obj), EV_TYPE_TRANSITION_ANIMATION, EvTransitionAnimationPriv))
30 typedef struct EvTransitionAnimationPriv EvTransitionAnimationPriv;
32 struct EvTransitionAnimationPriv {
33 EvTransitionEffect *effect;
34 cairo_surface_t *origin_surface;
35 cairo_surface_t *dest_surface;
46 G_DEFINE_TYPE (EvTransitionAnimation, ev_transition_animation, EV_TYPE_TIMELINE)
50 ev_transition_animation_init (EvTransitionAnimation *animation)
55 ev_transition_animation_set_property (GObject *object,
60 EvTransitionAnimationPriv *priv;
62 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (object);
67 g_object_unref (priv->effect);
69 priv->effect = g_value_dup_object (value);
71 case PROP_ORIGIN_SURFACE:
72 ev_transition_animation_set_origin_surface (EV_TRANSITION_ANIMATION (object),
73 g_value_get_pointer (value));
75 case PROP_DEST_SURFACE:
76 ev_transition_animation_set_dest_surface (EV_TRANSITION_ANIMATION (object),
77 g_value_get_pointer (value));
80 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
85 ev_transition_animation_get_property (GObject *object,
90 EvTransitionAnimationPriv *priv;
92 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (object);
96 g_value_set_object (value, priv->effect);
98 case PROP_ORIGIN_SURFACE:
99 g_value_set_pointer (value, priv->origin_surface);
101 case PROP_DEST_SURFACE:
102 g_value_set_pointer (value, priv->dest_surface);
105 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
110 ev_transition_animation_finalize (GObject *object)
112 EvTransitionAnimationPriv *priv;
114 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (object);
117 g_object_unref (priv->effect);
119 if (priv->origin_surface)
120 cairo_surface_destroy (priv->origin_surface);
122 if (priv->dest_surface)
123 cairo_surface_destroy (priv->dest_surface);
125 G_OBJECT_CLASS (ev_transition_animation_parent_class)->finalize (object);
129 ev_transition_animation_constructor (GType type,
130 guint n_construct_properties,
131 GObjectConstructParam *construct_params)
134 EvTransitionAnimationPriv *priv;
135 EvTransitionEffect *effect;
138 object = G_OBJECT_CLASS (ev_transition_animation_parent_class)->constructor (type,
139 n_construct_properties,
142 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (object);
143 effect = priv->effect;
145 g_object_get (effect, "duration", &duration, NULL);
146 ev_timeline_set_duration (EV_TIMELINE (object), duration * 1000);
152 ev_transition_animation_class_init (EvTransitionAnimationClass *klass)
154 GObjectClass *object_class = G_OBJECT_CLASS (klass);
156 object_class->set_property = ev_transition_animation_set_property;
157 object_class->get_property = ev_transition_animation_get_property;
158 object_class->finalize = ev_transition_animation_finalize;
159 object_class->constructor = ev_transition_animation_constructor;
161 g_object_class_install_property (object_class,
163 g_param_spec_object ("effect",
165 "Transition effect description",
166 EV_TYPE_TRANSITION_EFFECT,
167 G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE));
168 g_object_class_install_property (object_class,
170 g_param_spec_pointer ("origin-surface",
172 "Cairo surface from which the animation will happen",
174 g_object_class_install_property (object_class,
176 g_param_spec_pointer ("dest-surface",
177 "Destination surface",
178 "Cairo surface to which the animation will happen",
181 g_type_class_add_private (klass, sizeof (EvTransitionAnimationPriv));
185 paint_surface (cairo_t *cr,
186 cairo_surface_t *surface,
190 GdkRectangle page_area)
194 gdk_cairo_rectangle (cr, &page_area);
195 cairo_surface_set_device_offset (surface, x_offset, y_offset);
196 cairo_set_source_surface (cr, surface, 0, 0);
201 cairo_paint_with_alpha (cr, alpha);
208 ev_transition_animation_split (cairo_t *cr,
209 EvTransitionAnimation *animation,
210 EvTransitionEffect *effect,
212 GdkRectangle page_area)
214 EvTransitionAnimationPriv *priv;
215 EvTransitionEffectAlignment alignment;
216 EvTransitionEffectDirection direction;
219 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
220 width = page_area.width;
221 height = page_area.height;
223 g_object_get (effect,
224 "alignment", &alignment,
225 "direction", &direction,
228 if (direction == EV_TRANSITION_DIRECTION_INWARD) {
229 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
231 if (alignment == EV_TRANSITION_ALIGNMENT_HORIZONTAL) {
234 height * progress / 2,
236 height * (1 - progress));
239 width * progress / 2,
241 width * (1 - progress),
247 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
249 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
251 if (alignment == EV_TRANSITION_ALIGNMENT_HORIZONTAL) {
254 (height / 2) - (height * progress / 2),
259 (width / 2) - (width * progress / 2),
267 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
272 ev_transition_animation_blinds (cairo_t *cr,
273 EvTransitionAnimation *animation,
274 EvTransitionEffect *effect,
276 GdkRectangle page_area)
278 EvTransitionAnimationPriv *priv;
279 EvTransitionEffectAlignment alignment;
280 gint width, height, i;
282 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
283 width = page_area.width;
284 height = page_area.height;
286 g_object_get (effect,
287 "alignment", &alignment,
290 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
292 for (i = 0; i < N_BLINDS; i++) {
295 if (alignment == EV_TRANSITION_ALIGNMENT_HORIZONTAL) {
298 height / N_BLINDS * i,
300 height / N_BLINDS * progress);
303 width / N_BLINDS * i,
305 width / N_BLINDS * progress,
310 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
316 ev_transition_animation_box (cairo_t *cr,
317 EvTransitionAnimation *animation,
318 EvTransitionEffect *effect,
320 GdkRectangle page_area)
322 EvTransitionAnimationPriv *priv;
323 EvTransitionEffectDirection direction;
326 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
327 width = page_area.width;
328 height = page_area.height;
330 g_object_get (effect,
331 "direction", &direction,
334 if (direction == EV_TRANSITION_DIRECTION_INWARD) {
335 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
338 width * progress / 2,
339 height * progress / 2,
340 width * (1 - progress),
341 height * (1 - progress));
344 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
346 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
349 (width / 2) - (width * progress / 2),
350 (height / 2) - (height * progress / 2),
355 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
360 ev_transition_animation_wipe (cairo_t *cr,
361 EvTransitionAnimation *animation,
362 EvTransitionEffect *effect,
364 GdkRectangle page_area)
366 EvTransitionAnimationPriv *priv;
370 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
371 width = page_area.width;
372 height = page_area.height;
374 g_object_get (effect,
378 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
386 } else if (angle <= 90) {
390 height * (1 - progress),
393 } else if (angle <= 180) {
396 width * (1 - progress),
400 } else if (angle <= 270) {
410 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
414 ev_transition_animation_dissolve (cairo_t *cr,
415 EvTransitionAnimation *animation,
416 EvTransitionEffect *effect,
418 GdkRectangle page_area)
420 EvTransitionAnimationPriv *priv;
422 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
424 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
425 paint_surface (cr, priv->origin_surface, 0, 0, 1 - progress, page_area);
429 ev_transition_animation_push (cairo_t *cr,
430 EvTransitionAnimation *animation,
431 EvTransitionEffect *effect,
433 GdkRectangle page_area)
435 EvTransitionAnimationPriv *priv;
439 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
440 width = page_area.width;
441 height = page_area.height;
443 g_object_get (effect,
449 paint_surface (cr, priv->origin_surface, - (width * progress), 0, 0, page_area);
450 paint_surface (cr, priv->dest_surface, width * (1 - progress), 0, 0, page_area);
453 paint_surface (cr, priv->origin_surface, 0, - (height * progress), 0, page_area);
454 paint_surface (cr, priv->dest_surface, 0, height * (1 - progress), 0, page_area);
459 ev_transition_animation_cover (cairo_t *cr,
460 EvTransitionAnimation *animation,
461 EvTransitionEffect *effect,
463 GdkRectangle page_area)
465 EvTransitionAnimationPriv *priv;
469 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
470 width = page_area.width;
471 height = page_area.height;
473 g_object_get (effect,
477 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
481 paint_surface (cr, priv->dest_surface, width * (1 - progress), 0, 0, page_area);
484 paint_surface (cr, priv->dest_surface, 0, height * (1 - progress), 0, page_area);
489 ev_transition_animation_uncover (cairo_t *cr,
490 EvTransitionAnimation *animation,
491 EvTransitionEffect *effect,
493 GdkRectangle page_area)
495 EvTransitionAnimationPriv *priv;
499 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
500 width = page_area.width;
501 height = page_area.height;
503 g_object_get (effect,
507 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
511 paint_surface (cr, priv->origin_surface, - (width * progress), 0, 0, page_area);
514 paint_surface (cr, priv->origin_surface, 0, - (height * progress), 0, page_area);
519 ev_transition_animation_fade (cairo_t *cr,
520 EvTransitionAnimation *animation,
521 EvTransitionEffect *effect,
523 GdkRectangle page_area)
525 EvTransitionAnimationPriv *priv;
527 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
529 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
530 paint_surface (cr, priv->dest_surface, 0, 0, progress, page_area);
534 ev_transition_animation_paint (EvTransitionAnimation *animation,
536 GdkRectangle page_area)
538 EvTransitionAnimationPriv *priv;
539 EvTransitionEffectType type;
542 g_return_if_fail (EV_IS_TRANSITION_ANIMATION (animation));
544 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
546 if (!priv->dest_surface) {
547 /* animation is still not ready, paint the origin surface */
548 paint_surface (cr, priv->origin_surface, 0, 0, 0, page_area);
552 g_object_get (priv->effect, "type", &type, NULL);
553 progress = ev_timeline_get_progress (EV_TIMELINE (animation));
556 case EV_TRANSITION_EFFECT_REPLACE:
557 /* just paint the destination slide */
558 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
560 case EV_TRANSITION_EFFECT_SPLIT:
561 ev_transition_animation_split (cr, animation, priv->effect, progress, page_area);
563 case EV_TRANSITION_EFFECT_BLINDS:
564 ev_transition_animation_blinds (cr, animation, priv->effect, progress, page_area);
566 case EV_TRANSITION_EFFECT_BOX:
567 ev_transition_animation_box (cr, animation, priv->effect, progress, page_area);
569 case EV_TRANSITION_EFFECT_WIPE:
570 ev_transition_animation_wipe (cr, animation, priv->effect, progress, page_area);
572 case EV_TRANSITION_EFFECT_DISSOLVE:
573 ev_transition_animation_dissolve (cr, animation, priv->effect, progress, page_area);
575 case EV_TRANSITION_EFFECT_PUSH:
576 ev_transition_animation_push (cr, animation, priv->effect, progress, page_area);
578 case EV_TRANSITION_EFFECT_COVER:
579 ev_transition_animation_cover (cr, animation, priv->effect, progress, page_area);
581 case EV_TRANSITION_EFFECT_UNCOVER:
582 ev_transition_animation_uncover (cr, animation, priv->effect, progress, page_area);
584 case EV_TRANSITION_EFFECT_FADE:
585 ev_transition_animation_fade (cr, animation, priv->effect, progress, page_area);
588 GEnumValue *enum_value;
590 enum_value = g_enum_get_value (g_type_class_peek (EV_TYPE_TRANSITION_EFFECT_TYPE), type);
592 g_warning ("Unimplemented transition animation: '%s', "
593 "please post a bug report in Evince bugzilla "
594 "(http://bugzilla.gnome.org) with a testcase.",
595 enum_value->value_nick);
597 /* just paint the destination slide */
598 paint_surface (cr, priv->dest_surface, 0, 0, 0, page_area);
603 EvTransitionAnimation *
604 ev_transition_animation_new (EvTransitionEffect *effect)
606 g_return_val_if_fail (EV_IS_TRANSITION_EFFECT (effect), NULL);
608 return g_object_new (EV_TYPE_TRANSITION_ANIMATION,
614 ev_transition_animation_set_origin_surface (EvTransitionAnimation *animation,
615 cairo_surface_t *origin_surface)
617 EvTransitionAnimationPriv *priv;
618 cairo_surface_t *surface;
620 g_return_if_fail (EV_IS_TRANSITION_ANIMATION (animation));
622 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
624 if (priv->origin_surface == origin_surface)
627 surface = cairo_surface_reference (origin_surface);
629 if (priv->origin_surface)
630 cairo_surface_destroy (priv->origin_surface);
632 priv->origin_surface = surface;
633 g_object_notify (G_OBJECT (animation), "origin-surface");
635 if (priv->origin_surface && priv->dest_surface)
636 ev_timeline_start (EV_TIMELINE (animation));
640 ev_transition_animation_set_dest_surface (EvTransitionAnimation *animation,
641 cairo_surface_t *dest_surface)
643 EvTransitionAnimationPriv *priv;
644 cairo_surface_t *surface;
646 g_return_if_fail (EV_IS_TRANSITION_ANIMATION (animation));
648 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
650 if (priv->dest_surface == dest_surface)
653 surface = cairo_surface_reference (dest_surface);
655 if (priv->dest_surface)
656 cairo_surface_destroy (priv->dest_surface);
658 priv->dest_surface = surface;
659 g_object_notify (G_OBJECT (animation), "dest-surface");
661 if (priv->origin_surface && priv->dest_surface)
662 ev_timeline_start (EV_TIMELINE (animation));
666 ev_transition_animation_ready (EvTransitionAnimation *animation)
668 EvTransitionAnimationPriv *priv;
670 g_return_val_if_fail (EV_IS_TRANSITION_ANIMATION (animation), FALSE);
672 priv = EV_TRANSITION_ANIMATION_GET_PRIVATE (animation);
674 return (priv->origin_surface != NULL);