efl_animation: Add repeat_count property
authorJaehyun Cho <jae_hyun.cho@samsung.com>
Wed, 30 Aug 2017 11:20:15 +0000 (20:20 +0900)
committerJaehyun Cho <jae_hyun.cho@samsung.com>
Thu, 12 Oct 2017 12:03:49 +0000 (21:03 +0900)
21 files changed:
src/Makefile_Elementary.am
src/bin/elementary/Makefile.am
src/bin/elementary/test.c
src/bin/elementary/test_efl_anim_repeat.c [new file with mode: 0644]
src/lib/evas/Evas_Common.h
src/lib/evas/Evas_Internal.h
src/lib/evas/canvas/efl_animation.c
src/lib/evas/canvas/efl_animation.eo
src/lib/evas/canvas/efl_animation_alpha.c
src/lib/evas/canvas/efl_animation_group_parallel.c
src/lib/evas/canvas/efl_animation_group_sequential.c
src/lib/evas/canvas/efl_animation_object.c
src/lib/evas/canvas/efl_animation_object_group_parallel.c
src/lib/evas/canvas/efl_animation_object_group_parallel_private.h
src/lib/evas/canvas/efl_animation_object_group_sequential.c
src/lib/evas/canvas/efl_animation_object_group_sequential_private.h
src/lib/evas/canvas/efl_animation_object_private.h
src/lib/evas/canvas/efl_animation_private.h
src/lib/evas/canvas/efl_animation_rotate.c
src/lib/evas/canvas/efl_animation_scale.c
src/lib/evas/canvas/efl_animation_translate.c

index 5d7dc02..8d45eb3 100644 (file)
@@ -805,6 +805,7 @@ bin/elementary/test_efl_anim_group_parallel.c \
 bin/elementary/test_efl_anim_group_sequential.c \
 bin/elementary/test_efl_anim_event_anim.c \
 bin/elementary/test_efl_anim_pause.c \
+bin/elementary/test_efl_anim_repeat.c \
 bin/elementary/test_eio.c \
 bin/elementary/test_entry.c \
 bin/elementary/test_entry_anchor.c \
index 143800a..a9fb326 100644 (file)
@@ -40,6 +40,7 @@ test_efl_anim_group_parallel.c \
 test_efl_anim_group_sequential.c \
 test_efl_anim_event_anim.c \
 test_efl_anim_pause.c \
+test_efl_anim_repeat.c \
 test_application_server.c \
 test_bg.c \
 test_box.c \
index 8b9a098..e86b78d 100644 (file)
@@ -335,6 +335,7 @@ void test_efl_anim_group_parallel(void *data, Evas_Object *obj, void *event_info
 void test_efl_anim_group_sequential(void *data, Evas_Object *obj, void *event_info);
 void test_efl_anim_event_anim(void *data, Evas_Object *obj, void *event_info);
 void test_efl_anim_pause(void *data, Evas_Object *obj, void *event_info);
+void test_efl_anim_repeat(void *data, Evas_Object *obj, void *event_info);
 
 Evas_Object *win, *tbx; // TODO: refactoring
 void *tt;
@@ -817,6 +818,7 @@ add_tests:
    ADD_TEST(NULL, "Effects", "Efl Animation Group Sequential", test_efl_anim_group_sequential);
    ADD_TEST(NULL, "Effects", "Efl Animation Event Animation", test_efl_anim_event_anim);
    ADD_TEST(NULL, "Effects", "Efl Animation Pause", test_efl_anim_pause);
+   ADD_TEST(NULL, "Effects", "Efl Animation Repeat", test_efl_anim_repeat);
 
    //------------------------------//
    ADD_TEST(NULL, "Edje External", "ExtButton", test_external_button);
diff --git a/src/bin/elementary/test_efl_anim_repeat.c b/src/bin/elementary/test_efl_anim_repeat.c
new file mode 100644 (file)
index 0000000..912b239
--- /dev/null
@@ -0,0 +1,159 @@
+#ifdef HAVE_CONFIG_H
+# include "elementary_config.h"
+#endif
+#include <Elementary.h>
+
+typedef struct _App_Data
+{
+   Efl_Animation        *show_anim;
+   Efl_Animation        *hide_anim;
+   Efl_Animation_Object *anim_obj;
+
+   Evas_Object          *repeat_count_spin;
+
+   Eina_Bool             is_btn_visible;
+} App_Data;
+
+static void
+_anim_started_cb(void *data EINA_UNUSED, const Efl_Event *event EINA_UNUSED)
+{
+   printf("Animation has been started!\n");
+}
+
+static void
+_anim_ended_cb(void *data, const Efl_Event *event EINA_UNUSED)
+{
+   App_Data *ad = data;
+
+   printf("Animation has been ended!\n");
+
+   elm_object_disabled_set(ad->repeat_count_spin, EINA_FALSE);
+
+   ad->anim_obj = NULL;
+}
+
+static void
+_anim_running_cb(void *data EINA_UNUSED, const Efl_Event *event)
+{
+   Efl_Animation_Object_Running_Event_Info *event_info = event->info;
+   double progress = event_info->progress;
+   printf("Animation is running! Current progress(%lf)\n", progress);
+}
+
+static void
+_start_btn_clicked_cb(void *data, Evas_Object *obj, void *event_info EINA_UNUSED)
+{
+   App_Data *ad = data;
+
+   if (ad->anim_obj)
+     efl_animation_object_cancel(ad->anim_obj);
+
+   ad->is_btn_visible = !(ad->is_btn_visible);
+
+   int repeat_count = elm_spinner_value_get(ad->repeat_count_spin);
+   elm_object_disabled_set(ad->repeat_count_spin, EINA_TRUE);
+
+   if (ad->is_btn_visible)
+     {
+        //Set animation repeat count
+        efl_animation_repeat_count_set(ad->show_anim, repeat_count);
+
+        //Create Animation Object from Animation
+        ad->anim_obj = efl_animation_object_create(ad->show_anim);
+        elm_object_text_set(obj, "Start Alpha Animation from 1.0 to 0.0");
+     }
+   else
+     {
+        //Set animation repeat count
+        efl_animation_repeat_count_set(ad->hide_anim, repeat_count);
+
+        //Create Animation Object from Animation
+        ad->anim_obj = efl_animation_object_create(ad->hide_anim);
+        elm_object_text_set(obj, "Start Alpha Animation from 0.0 to 1.0");
+     }
+
+   //Register callback called when animation starts
+   efl_event_callback_add(ad->anim_obj, EFL_ANIMATION_OBJECT_EVENT_STARTED, _anim_started_cb, NULL);
+
+   //Register callback called when animation ends
+   efl_event_callback_add(ad->anim_obj, EFL_ANIMATION_OBJECT_EVENT_ENDED, _anim_ended_cb, ad);
+
+   //Register callback called while animation is executed
+   efl_event_callback_add(ad->anim_obj, EFL_ANIMATION_OBJECT_EVENT_RUNNING, _anim_running_cb, NULL);
+
+   //Let Animation Object start animation
+   efl_animation_object_start(ad->anim_obj);
+}
+
+static void
+_win_del_cb(void *data, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED)
+{
+   App_Data *ad = data;
+   free(ad);
+}
+
+void
+test_efl_anim_repeat(void *data EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED)
+{
+   App_Data *ad = calloc(1, sizeof(App_Data));
+   if (!ad) return;
+
+   Evas_Object *win = elm_win_add(NULL, "Efl Animation Repeat", ELM_WIN_BASIC);
+   elm_win_title_set(win, "Efl Animation Repeat");
+   elm_win_autodel_set(win, EINA_TRUE);
+   evas_object_smart_callback_add(win, "delete,request", _win_del_cb, ad);
+
+   //Button to be animated
+   Evas_Object *btn = elm_button_add(win);
+   elm_object_text_set(btn, "Button");
+   evas_object_size_hint_weight_set(btn, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
+   evas_object_resize(btn, 200, 200);
+   evas_object_move(btn, 100, 50);
+   evas_object_show(btn);
+
+   //Show Animation
+   Efl_Animation *show_anim = efl_add(EFL_ANIMATION_ALPHA_CLASS, NULL);
+   efl_animation_alpha_set(show_anim, 0.0, 1.0);
+   efl_animation_duration_set(show_anim, 1.0);
+   efl_animation_target_set(show_anim, btn);
+   efl_animation_final_state_keep_set(show_anim, EINA_TRUE);
+
+   //Hide Animation
+   Efl_Animation *hide_anim = efl_add(EFL_ANIMATION_ALPHA_CLASS, NULL);
+   efl_animation_alpha_set(hide_anim, 1.0, 0.0);
+   efl_animation_duration_set(hide_anim, 1.0);
+   efl_animation_target_set(hide_anim, btn);
+   efl_animation_final_state_keep_set(hide_anim, EINA_TRUE);
+
+
+   //Button to start animation
+   Evas_Object *start_btn = elm_button_add(win);
+   elm_object_text_set(start_btn, "Start Alpha Animation from 1.0 to 0.0");
+   evas_object_smart_callback_add(start_btn, "clicked", _start_btn_clicked_cb, ad);
+   evas_object_size_hint_weight_set(start_btn, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
+   evas_object_resize(start_btn, 200, 50);
+   evas_object_move(start_btn, 100, 300);
+   evas_object_show(start_btn);
+
+   //Spinner to set animation repeat count
+   Evas_Object *repeat_count_spin = elm_spinner_add(win);
+   elm_spinner_label_format_set(repeat_count_spin, "Repeat Count: %d");
+   elm_spinner_editable_set(repeat_count_spin, EINA_FALSE);
+   elm_spinner_min_max_set(repeat_count_spin, 0, 3);
+   elm_spinner_value_set(repeat_count_spin, 0);
+   evas_object_resize(repeat_count_spin, 200, 50);
+   evas_object_move(repeat_count_spin, 100, 350);
+   evas_object_show(repeat_count_spin);
+
+
+   //Initialize App Data
+   ad->show_anim = show_anim;
+   ad->hide_anim = hide_anim;
+   ad->anim_obj = NULL;
+   ad->repeat_count_spin = repeat_count_spin;
+   ad->is_btn_visible = EINA_TRUE;
+
+
+   evas_object_resize(win, 400, 450);
+   evas_object_show(win);
+}
index 704ef81..356edc7 100644 (file)
@@ -3450,6 +3450,9 @@ struct _Efl_Animation_Object_Running_Event_Info
 #define EFL_ANIMATION_GROUP_DURATION_NONE -1
 #define EFL_ANIMATION_OBJECT_GROUP_DURATION_NONE -1
 
+#define EFL_ANIMATION_REPEAT_INFINITE -1
+#define EFL_ANIMATION_OBJECT_REPEAT_INFINITE -1
+
 /**
  * @}
  */
index 721eba6..45057bd 100644 (file)
@@ -98,6 +98,9 @@ EOAPI void efl_animation_object_duration_only_set(Eo *obj, double duration);
 EOAPI void efl_animation_object_total_duration_set(Eo *obj, double total_duration);
 EOAPI double efl_animation_object_total_duration_get(const Eo *obj);
 
+EOAPI void efl_animation_object_repeat_count_set(Eo *obj, int count);
+EOAPI int efl_animation_object_repeat_count_get(const Eo *obj);
+
 EWAPI extern const Efl_Event_Description _EFL_ANIMATION_OBJECT_EVENT_PRE_STARTED;
 #define EFL_ANIMATION_OBJECT_EVENT_PRE_STARTED (&(_EFL_ANIMATION_OBJECT_EVENT_PRE_STARTED))
 /* Efl.Animation.Object END */
index 2246684..3ab7434 100644 (file)
@@ -127,9 +127,33 @@ _efl_animation_object_create(Eo *eo_obj, Efl_Animation_Data *pd EINA_UNUSED)
    double total_duration = efl_animation_total_duration_get(eo_obj);
    efl_animation_object_total_duration_set(anim_obj, total_duration);
 
+   int repeat_count = efl_animation_repeat_count_get(eo_obj);
+   efl_animation_object_repeat_count_set(anim_obj, repeat_count);
+
    return anim_obj;
 }
 
+EOLIAN static void
+_efl_animation_repeat_count_set(Eo *eo_obj,
+                                Efl_Animation_Data *pd,
+                                int count)
+{
+   EFL_ANIMATION_CHECK_OR_RETURN(eo_obj);
+
+   //EFL_ANIMATION_REPEAT_INFINITE repeats animation infinitely
+   if ((count < 0) && (count != EFL_ANIMATION_REPEAT_INFINITE)) return;
+
+   pd->repeat_count = count;
+}
+
+EOLIAN static int
+_efl_animation_repeat_count_get(Eo *eo_obj, Efl_Animation_Data *pd)
+{
+   EFL_ANIMATION_CHECK_OR_RETURN(eo_obj, 0);
+
+   return pd->repeat_count;
+}
+
 EOLIAN static Efl_Object *
 _efl_animation_efl_object_constructor(Eo *eo_obj,
                                       Efl_Animation_Data *pd)
@@ -140,6 +164,8 @@ _efl_animation_efl_object_constructor(Eo *eo_obj,
 
    pd->duration = 0.0;
 
+   pd->repeat_count = 0;
+
    pd->is_deleted = EINA_FALSE;
    pd->keep_final_state = EINA_FALSE;
 
index be899c4..d071100 100644 (file)
@@ -47,6 +47,15 @@ class Efl.Animation (Efl.Object)
             total_duration: double; [[Total duration value.]]
          }
       }
+      @property repeat_count {
+         set {
+         }
+         get {
+         }
+         values {
+            count: int; [[Repeat count. EFL_ANIMATION_REPEAT_INFINITE repeats animation infinitely.]]
+         }
+      }
       is_deleted @protected {
          return: bool; [[$true if animation is deleted, $false otherwise.]]
       }
index deddafe..763788d 100644 (file)
@@ -47,6 +47,9 @@ _efl_animation_alpha_efl_animation_object_create(Eo *eo_obj,
    double total_duration = efl_animation_total_duration_get(eo_obj);
    efl_animation_object_total_duration_set(anim_obj, total_duration);
 
+   int repeat_count = efl_animation_repeat_count_get(eo_obj);
+   efl_animation_object_repeat_count_set(anim_obj, repeat_count);
+
    efl_animation_object_alpha_set(anim_obj, pd->from.alpha, pd->to.alpha);
 
    return anim_obj;
index 9380731..9c02c8c 100644 (file)
@@ -58,6 +58,11 @@ _efl_animation_group_parallel_efl_animation_total_duration_get(Eo *eo_obj,
    EINA_LIST_FOREACH(animations, l, anim)
      {
         double child_total_duration = efl_animation_total_duration_get(anim);
+
+        int child_repeat_count = efl_animation_repeat_count_get(anim);
+        if (child_repeat_count > 0)
+          child_total_duration *= (child_repeat_count + 1);
+
         if (child_total_duration > total_duration)
           total_duration = child_total_duration;
      }
@@ -97,6 +102,9 @@ _efl_animation_group_parallel_efl_animation_object_create(Eo *eo_obj,
    double total_duration = efl_animation_total_duration_get(eo_obj);
    efl_animation_object_total_duration_set(group_anim_obj, total_duration);
 
+   int repeat_count = efl_animation_repeat_count_get(eo_obj);
+   efl_animation_object_repeat_count_set(group_anim_obj, repeat_count);
+
    return group_anim_obj;
 }
 
index de09fcf..b29599b 100644 (file)
@@ -52,7 +52,13 @@ _efl_animation_group_sequential_efl_animation_total_duration_get(Eo *eo_obj,
    Efl_Animation *anim;
    EINA_LIST_FOREACH(animations, l, anim)
      {
-        total_duration += efl_animation_total_duration_get(anim);
+        double child_total_duration = efl_animation_total_duration_get(anim);
+
+        int child_repeat_count = efl_animation_repeat_count_get(anim);
+        if (child_repeat_count > 0)
+          child_total_duration *= (child_repeat_count + 1);
+
+        total_duration += child_total_duration;
      }
    return total_duration;
 }
@@ -90,6 +96,9 @@ _efl_animation_group_sequential_efl_animation_object_create(Eo *eo_obj,
    double total_duration = efl_animation_total_duration_get(eo_obj);
    efl_animation_object_total_duration_set(group_anim_obj, total_duration);
 
+   int repeat_count = efl_animation_repeat_count_get(eo_obj);
+   efl_animation_object_repeat_count_set(group_anim_obj, repeat_count);
+
    return group_anim_obj;
 }
 
index 1202d5e..619af19 100644 (file)
@@ -105,6 +105,28 @@ _efl_animation_object_total_duration_get(Eo *eo_obj, Efl_Animation_Object_Data *
    return pd->total_duration;
 }
 
+EOLIAN static void
+_efl_animation_object_repeat_count_set(Eo *eo_obj,
+                                       Efl_Animation_Object_Data *pd,
+                                       int count)
+{
+   EFL_ANIMATION_OBJECT_CHECK_OR_RETURN(eo_obj);
+
+   //EFL_ANIMATION_OBJECT_REPEAT_INFINITE repeats animation infinitely
+   if ((count < 0) && (count != EFL_ANIMATION_OBJECT_REPEAT_INFINITE)) return;
+
+   pd->repeat_count = count;
+}
+
+EOLIAN static int
+_efl_animation_object_repeat_count_get(const Eo *eo_obj,
+                                       Efl_Animation_Object_Data *pd)
+{
+   EFL_ANIMATION_OBJECT_CHECK_OR_RETURN((Eo *)eo_obj, 0);
+
+   return pd->repeat_count;
+}
+
 EOLIAN static Eina_Bool
 _efl_animation_object_is_deleted(Eo *eo_obj,
                                  Efl_Animation_Object_Data *pd)
@@ -267,6 +289,21 @@ _animator_cb(void *data)
      return ECORE_CALLBACK_RENEW;
 
 end:
+   //Repeat animation
+   if ((pd->repeat_count == EFL_ANIMATION_OBJECT_REPEAT_INFINITE) ||
+       (pd->remaining_repeat_count > 0))
+     {
+        if (pd->remaining_repeat_count > 0)
+          pd->remaining_repeat_count--;
+
+        pd->time.begin = ecore_loop_time_get();
+        pd->paused_time = 0.0;
+
+        efl_animation_object_target_state_reset(eo_obj);
+
+        return ECORE_CALLBACK_RENEW;
+     }
+
    pd->is_ended = EINA_TRUE;
    pd->animator = NULL;
 
@@ -296,6 +333,8 @@ _start(Eo *eo_obj, Efl_Animation_Object_Data *pd)
 
    pd->paused_time = 0.0;
 
+   pd->remaining_repeat_count = pd->repeat_count;
+
    ecore_animator_del(pd->animator);
    pd->animator = NULL;
 
@@ -414,6 +453,8 @@ _efl_animation_object_efl_object_constructor(Eo *eo_obj,
    pd->duration = 0.0;
    pd->total_duration = 0.0;
 
+   pd->repeat_count = 0;
+
    pd->auto_del = EINA_TRUE;
    pd->is_deleted = EINA_FALSE;
    pd->is_cancelled = EINA_FALSE;
@@ -465,6 +506,9 @@ EOAPI EFL_VOID_FUNC_BODYV(efl_animation_object_duration_only_set, EFL_FUNC_CALL(
 EOAPI EFL_VOID_FUNC_BODYV(efl_animation_object_total_duration_set, EFL_FUNC_CALL(total_duration), double total_duration);
 EOAPI EFL_FUNC_BODY_CONST(efl_animation_object_total_duration_get, double, 0);
 
+EOAPI EFL_VOID_FUNC_BODYV(efl_animation_object_repeat_count_set, EFL_FUNC_CALL(count), int count);
+EOAPI EFL_FUNC_BODY_CONST(efl_animation_object_repeat_count_get, int, 0);
+
 #define EFL_ANIMATION_OBJECT_EXTRA_OPS \
    EFL_OBJECT_OP_FUNC(efl_animation_object_target_set, _efl_animation_object_target_set), \
    EFL_OBJECT_OP_FUNC(efl_animation_object_target_get, _efl_animation_object_target_get), \
@@ -474,7 +518,9 @@ EOAPI EFL_FUNC_BODY_CONST(efl_animation_object_total_duration_get, double, 0);
    EFL_OBJECT_OP_FUNC(efl_animation_object_duration_get, _efl_animation_object_duration_get), \
    EFL_OBJECT_OP_FUNC(efl_animation_object_duration_only_set, _efl_animation_object_duration_only_set), \
    EFL_OBJECT_OP_FUNC(efl_animation_object_total_duration_set, _efl_animation_object_total_duration_set), \
-   EFL_OBJECT_OP_FUNC(efl_animation_object_total_duration_get, _efl_animation_object_total_duration_get)
+   EFL_OBJECT_OP_FUNC(efl_animation_object_total_duration_get, _efl_animation_object_total_duration_get), \
+   EFL_OBJECT_OP_FUNC(efl_animation_object_repeat_count_set, _efl_animation_object_repeat_count_set), \
+   EFL_OBJECT_OP_FUNC(efl_animation_object_repeat_count_get, _efl_animation_object_repeat_count_get)
 
 EWAPI const Efl_Event_Description _EFL_ANIMATION_OBJECT_EVENT_PRE_STARTED =
    EFL_EVENT_DESCRIPTION("pre_started");
index a2f4772..7e23b73 100644 (file)
@@ -1,8 +1,62 @@
 #include "efl_animation_object_group_parallel_private.h"
 
+/* Add member object data and append the data to the member object data list.
+ * The member object data contains the repeated count of the member object.
+ */
+static void
+_member_anim_obj_data_add(Efl_Animation_Object_Group_Parallel_Data *pd,
+                          Efl_Animation_Object *anim_obj,
+                          int repeated_count)
+{
+   Member_Object_Data *member_anim_obj_data =
+      calloc(1, sizeof(Member_Object_Data));
+
+   if (!member_anim_obj_data) return;
+
+   member_anim_obj_data->anim_obj = anim_obj;
+   member_anim_obj_data->repeated_count = repeated_count;
+
+   pd->member_anim_obj_data_list =
+      eina_list_append(pd->member_anim_obj_data_list, member_anim_obj_data);
+}
+
+/* Find the member object data which contains the repeated count of the member
+ * object. */
+static Member_Object_Data *
+_member_anim_obj_data_find(Efl_Animation_Object_Group_Parallel_Data *pd,
+                           Efl_Animation_Object *anim_obj)
+{
+   Eina_List *l;
+   Member_Object_Data *member_anim_obj_data = NULL;
+   EINA_LIST_FOREACH(pd->member_anim_obj_data_list, l, member_anim_obj_data)
+     {
+        if (member_anim_obj_data->anim_obj == anim_obj)
+          break;
+     }
+
+   return member_anim_obj_data;
+}
+
+/* Delete member object data and remove the data from the member object data
+ * list.
+ * The member object data contains the repeated count of the member object.
+ */
+static void
+_member_anim_obj_data_del(Efl_Animation_Object_Group_Parallel_Data *pd,
+                          Efl_Animation_Object *anim_obj)
+{
+   Member_Object_Data *member_anim_obj_data = _member_anim_obj_data_find(pd, anim_obj);
+   if (member_anim_obj_data)
+     {
+        pd->member_anim_obj_data_list =
+           eina_list_remove(pd->member_anim_obj_data_list, member_anim_obj_data);
+        free(member_anim_obj_data);
+     }
+}
+
 EOLIAN static void
 _efl_animation_object_group_parallel_efl_animation_object_group_object_add(Eo *eo_obj,
-                                                                           Efl_Animation_Object_Group_Parallel_Data *pd EINA_UNUSED,
+                                                                           Efl_Animation_Object_Group_Parallel_Data *pd,
                                                                            Efl_Animation_Object *anim_obj)
 {
    EFL_ANIMATION_OBJECT_GROUP_PARALLEL_CHECK_OR_RETURN(eo_obj);
@@ -11,6 +65,10 @@ _efl_animation_object_group_parallel_efl_animation_object_group_object_add(Eo *e
 
    efl_animation_object_group_object_add(efl_super(eo_obj, MY_CLASS), anim_obj);
 
+   /* Add member object data and append the data to the member object data
+    * list. */
+   _member_anim_obj_data_add(pd, anim_obj, 0);
+
    /* Total duration is calculated in
     * efl_animation_object_total_duration_get() based on the current group
     * animation object list.
@@ -31,6 +89,10 @@ _efl_animation_object_group_parallel_efl_animation_object_group_object_del(Eo *e
 
    efl_animation_object_group_object_del(efl_super(eo_obj, MY_CLASS), anim_obj);
 
+   /* Delete member object data and remove the data from the member object
+    * data list. */
+   _member_anim_obj_data_del(pd, anim_obj);
+
    /* Total duration is calculated in
     * efl_animation_object_total_duration_get() based on the current group
     * animation object list.
@@ -56,15 +118,47 @@ _efl_animation_object_group_parallel_efl_animation_object_total_duration_get(Eo
      {
         double child_total_duration =
            efl_animation_object_total_duration_get(anim_obj);
+
+        int child_repeat_count =
+           efl_animation_object_repeat_count_get(anim_obj);
+        if (child_repeat_count > 0)
+          child_total_duration *= (child_repeat_count + 1);
+
         if (child_total_duration > total_duration)
           total_duration = child_total_duration;
      }
    return total_duration;
 }
 
+//Set how many times the given object has been repeated.
+static void
+_repeated_count_set(Efl_Animation_Object_Group_Parallel_Data *pd,
+                    Efl_Animation_Object *anim_obj,
+                    int repeated_count)
+{
+
+   Member_Object_Data *member_anim_obj_data =
+      _member_anim_obj_data_find(pd, anim_obj);
+   if (!member_anim_obj_data) return;
+
+   member_anim_obj_data->repeated_count = repeated_count;
+}
+
+//Get how many times the given object has been repeated.
+static int
+_repeated_count_get(Efl_Animation_Object_Group_Parallel_Data *pd,
+                    Efl_Animation_Object *anim_obj)
+{
+   Member_Object_Data *member_anim_obj_data =
+      _member_anim_obj_data_find(pd, anim_obj);
+   if (!member_anim_obj_data) return 0;
+
+   return member_anim_obj_data->repeated_count;
+}
+
 EOLIAN static void
 _efl_animation_object_group_parallel_efl_animation_object_progress_set(Eo *eo_obj,
-                                                                       Efl_Animation_Object_Group_Parallel_Data *pd EINA_UNUSED,
+                                                                       Efl_Animation_Object_Group_Parallel_Data *pd,
                                                                        double progress)
 {
    if ((progress < 0.0) || (progress > 1.0)) return;
@@ -89,9 +183,33 @@ _efl_animation_object_group_parallel_efl_animation_object_progress_set(Eo *eo_ob
           anim_obj_progress = 1.0;
         else
           {
-             anim_obj_progress = elapsed_time / total_duration;
+             //If object is repeated, then recalculate progress.
+             int repeated_count = _repeated_count_get(pd, anim_obj);
+             if (repeated_count > 0)
+               anim_obj_progress = (elapsed_time - (total_duration * repeated_count)) / total_duration;
+             else
+               anim_obj_progress = elapsed_time / total_duration;
+
              if (anim_obj_progress > 1.0)
                anim_obj_progress = 1.0;
+
+             //Animation has been finished.
+             if (anim_obj_progress == 1.0)
+               {
+                  /* If object is finished and it should be repeated, then
+                   * increate the repeated count to recalculate progress. */
+                  int repeat_count =
+                     efl_animation_object_repeat_count_get(anim_obj);
+                  if (repeat_count > 0)
+                    {
+                       int repeated_count = _repeated_count_get(pd, anim_obj);
+                       if (repeated_count < repeat_count)
+                         {
+                            repeated_count++;
+                            _repeated_count_set(pd, anim_obj, repeated_count);
+                         }
+                    }
+               }
           }
 
         efl_animation_object_progress_set(anim_obj, anim_obj_progress);
index 85ab1f2..8ccfe5d 100644 (file)
 #define EFL_ANIMATION_OBJECT_GROUP_PARALLEL_DATA_GET(o, pd) \
    Efl_Animation_Object_Group_Parallel_Data *pd = efl_data_scope_get(o, EFL_ANIMATION_OBJECT_GROUP_PARALLEL_CLASS)
 
+typedef struct _Member_Object_Data
+{
+   Efl_Animation_Object *anim_obj;
+
+   int                   repeated_count; /* How many times object has been
+                                          * repeated. */
+} Member_Object_Data;
+
 typedef struct _Efl_Animation_Object_Group_Parallel_Data
 {
+   Eina_List *member_anim_obj_data_list; //List of Member_Object_Data
 } Efl_Animation_Object_Group_Parallel_Data;
index 98f6b13..9dcbd2a 100644 (file)
@@ -1,5 +1,61 @@
 #include "efl_animation_object_group_sequential_private.h"
 
+/* Add member object data and append the data to the member object data list.
+ * The member object data contains the repeated count of the member object.
+ */
+static void
+_member_anim_obj_data_add(Efl_Animation_Object_Group_Sequential_Data *pd,
+                          Efl_Animation_Object *anim_obj,
+                          int repeated_count)
+{
+   Member_Object_Data *member_anim_obj_data =
+      calloc(1, sizeof(Member_Object_Data));
+
+   if (!member_anim_obj_data) return;
+
+   member_anim_obj_data->anim_obj = anim_obj;
+   member_anim_obj_data->repeated_count = repeated_count;
+
+   pd->member_anim_obj_data_list =
+      eina_list_append(pd->member_anim_obj_data_list, member_anim_obj_data);
+}
+
+/* Find the member object data which contains the repeated count of the member
+ * object. */
+static Member_Object_Data *
+_member_anim_obj_data_find(Efl_Animation_Object_Group_Sequential_Data *pd,
+                           Efl_Animation_Object *anim_obj)
+{
+   Eina_List *l;
+   Member_Object_Data *member_anim_obj_data = NULL;
+   EINA_LIST_FOREACH(pd->member_anim_obj_data_list, l, member_anim_obj_data)
+     {
+        if (member_anim_obj_data->anim_obj == anim_obj)
+          break;
+     }
+
+   return member_anim_obj_data;
+}
+
+/* Delete member object data and remove the data from the member object data
+ * list.
+ * The member object data contains the repeated count of the member object.
+ */
+static void
+_member_anim_obj_data_del(Efl_Animation_Object_Group_Sequential_Data *pd,
+                          Efl_Animation_Object *anim_obj)
+{
+   Member_Object_Data *member_anim_obj_data =
+      _member_anim_obj_data_find(pd, anim_obj);
+   if (member_anim_obj_data)
+     {
+        pd->member_anim_obj_data_list =
+           eina_list_remove(pd->member_anim_obj_data_list,
+                            member_anim_obj_data);
+        free(member_anim_obj_data);
+     }
+}
+
 EOLIAN static void
 _efl_animation_object_group_sequential_efl_animation_object_group_object_add(Eo *eo_obj,
                                                                              Efl_Animation_Object_Group_Sequential_Data *pd EINA_UNUSED,
@@ -11,6 +67,10 @@ _efl_animation_object_group_sequential_efl_animation_object_group_object_add(Eo
 
    efl_animation_object_group_object_add(efl_super(eo_obj, MY_CLASS), anim_obj);
 
+   /* Add member object data and append the data to the member object data
+    * list. */
+   _member_anim_obj_data_add(pd, anim_obj, 0);
+
    /* Total duration is calculated in
     * efl_animation_object_total_duration_get() based on the current group
     * animation object list.
@@ -31,6 +91,10 @@ _efl_animation_object_group_sequential_efl_animation_object_group_object_del(Eo
 
    efl_animation_object_group_object_del(efl_super(eo_obj, MY_CLASS), anim_obj);
 
+   /* Delete member object data and remove the data from the member object
+    * data list. */
+   _member_anim_obj_data_del(pd, anim_obj);
+
    /* Total duration is calculated in
     * efl_animation_object_total_duration_get() based on the current group
     * animation object list.
@@ -54,11 +118,44 @@ _efl_animation_object_group_sequential_efl_animation_object_total_duration_get(E
    Efl_Animation *anim_obj;
    EINA_LIST_FOREACH(anim_objs, l, anim_obj)
      {
-        total_duration += efl_animation_object_total_duration_get(anim_obj);
+        double child_total_duration =
+           efl_animation_object_total_duration_get(anim_obj);
+
+        int child_repeat_count =
+           efl_animation_object_repeat_count_get(anim_obj);
+        if (child_repeat_count > 0)
+          child_total_duration *= (child_repeat_count + 1);
+
+        total_duration += child_total_duration;
      }
    return total_duration;
 }
 
+//Set how many times the given object has been repeated.
+static void
+_repeated_count_set(Efl_Animation_Object_Group_Sequential_Data *pd,
+                    Efl_Animation_Object *anim_obj,
+                    int repeated_count)
+{
+
+   Member_Object_Data *member_anim_obj_data = _member_anim_obj_data_find(pd, anim_obj);
+   if (!member_anim_obj_data) return;
+
+   member_anim_obj_data->repeated_count = repeated_count;
+}
+
+//Get how many times the given object has been repeated.
+static int
+_repeated_count_get(Efl_Animation_Object_Group_Sequential_Data *pd,
+                    Efl_Animation_Object *anim_obj)
+{
+   Member_Object_Data *member_anim_obj_data =
+      _member_anim_obj_data_find(pd, anim_obj);
+   if (!member_anim_obj_data) return 0;
+
+   return member_anim_obj_data->repeated_count;
+}
+
 EOLIAN static void
 _efl_animation_object_group_sequential_efl_animation_object_progress_set(Eo *eo_obj,
                                                                          Efl_Animation_Object_Group_Sequential_Data *pd EINA_UNUSED,
@@ -92,10 +189,33 @@ _efl_animation_object_group_sequential_efl_animation_object_progress_set(Eo *eo_
           anim_obj_progress = 1.0;
         else
           {
+             //If object is repeated, then recalculate progress.
+             int repeated_count = _repeated_count_get(pd, anim_obj);
+             if (repeated_count > 0)
+               sum_prev_total_duration += (total_duration * repeated_count);
+
              anim_obj_progress =
                 (elapsed_time - sum_prev_total_duration) / total_duration;
              if (anim_obj_progress > 1.0)
                anim_obj_progress = 1.0;
+
+             //Animation has been finished.
+             if (anim_obj_progress == 1.0)
+               {
+                  /* If object is finished and it should be repeated, then
+                   * increate the repeated count to recalculate progress. */
+                  int repeat_count =
+                     efl_animation_object_repeat_count_get(anim_obj);
+                  if (repeat_count > 0)
+                    {
+                       int repeated_count = _repeated_count_get(pd, anim_obj);
+                       if (repeated_count < repeat_count)
+                         {
+                            repeated_count++;
+                            _repeated_count_set(pd, anim_obj, repeated_count);
+                         }
+                    }
+               }
           }
 
         //Update the sum of the previous objects' total durations
index fc68a14..aa6f989 100644 (file)
 #define EFL_ANIMATION_OBJECT_GROUP_SEQUENTIAL_DATA_GET(o, pd) \
    Efl_Animation_Object_Group_Sequential_Data *pd = efl_data_scope_get(o, EFL_ANIMATION_OBJECT_GROUP_SEQUENTIAL_CLASS)
 
+typedef struct _Member_Object_Data
+{
+   Efl_Animation_Object *anim_obj;
+
+   int                   repeated_count; /* How many times object has been
+                                          * repeated. */
+} Member_Object_Data;
+
 typedef struct _Efl_Animation_Object_Group_Sequential_Data
 {
+   Eina_List   *member_anim_obj_data_list; //List of Member_Object_Data
 } Efl_Animation_Object_Group_Sequential_Data;
index 52bacad..f82d24f 100644 (file)
@@ -34,6 +34,9 @@ typedef struct _Efl_Animation_Object_Data
    double             total_duration;
    double             paused_time;
 
+   int                repeat_count;
+   int                remaining_repeat_count;
+
    Eina_Bool          auto_del : 1;
    Eina_Bool          is_deleted : 1;
    Eina_Bool          is_started : 1;
index 0c3cea1..49e635c 100644 (file)
@@ -12,6 +12,8 @@ typedef struct _Efl_Animation_Data
    double             duration;
    double             total_duration;
 
+   int                repeat_count;
+
    Eina_Bool          is_deleted : 1;
    Eina_Bool          keep_final_state : 1;
 } Efl_Animation_Data;
index 2d6399f..362464c 100644 (file)
@@ -195,6 +195,9 @@ _efl_animation_rotate_efl_animation_object_create(Eo *eo_obj,
    double total_duration = efl_animation_total_duration_get(eo_obj);
    efl_animation_object_total_duration_set(anim_obj, total_duration);
 
+   int repeat_count = efl_animation_repeat_count_get(eo_obj);
+   efl_animation_object_repeat_count_set(anim_obj, repeat_count);
+
    if (pd->use_rel_pivot)
      {
         efl_animation_object_rotate_set(anim_obj,
index ec04c45..dad906e 100644 (file)
@@ -221,6 +221,9 @@ _efl_animation_scale_efl_animation_object_create(Eo *eo_obj,
    double total_duration = efl_animation_total_duration_get(eo_obj);
    efl_animation_object_total_duration_set(anim_obj, total_duration);
 
+   int repeat_count = efl_animation_repeat_count_get(eo_obj);
+   efl_animation_object_repeat_count_set(anim_obj, repeat_count);
+
    if (pd->use_rel_pivot)
      {
         efl_animation_object_scale_set(anim_obj,
index 1b0f0e3..5623fb0 100644 (file)
@@ -163,6 +163,9 @@ _efl_animation_translate_efl_animation_object_create(Eo *eo_obj,
    double total_duration = efl_animation_total_duration_get(eo_obj);
    efl_animation_object_total_duration_set(anim_obj, total_duration);
 
+   int repeat_count = efl_animation_repeat_count_get(eo_obj);
+   efl_animation_object_repeat_count_set(anim_obj, repeat_count);
+
    if (pd->use_rel_move)
      {
         efl_animation_object_translate_set(anim_obj,