default_backend: implements frame_duration to proceed each frame at a different speed
[platform/core/uifw/libpui.git] / backends / system / default_ani_swupdatedone.c
1 /*
2  * Copyright © 2019 Samsung Electronics co., Ltd. All Rights Reserved.
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining
5  * a copy of this software and associated documentation files (the
6  * "Software"), to deal in the Software without restriction, including
7  * without limitation the rights to use, copy, modify, merge, publish,
8  * distribute, sublicense, and/or sell copies of the Software, and to
9  * permit persons to whom the Software is furnished to do so, subject to
10  * the following conditions:
11  *
12  * The above copyright notice and this permission notice (including the
13  * next paragraph) shall be included in all copies or substantial
14  * portions of the Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19  * NONINFRINGEMENT.  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
20  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
21  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
22  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
23  * SOFTWARE.
24  */
25
26 #include "../default_backend.h"
27
28 static void
29 _ani_backend_swupdatedone_get_led_rgb(default_frame_info_t *frame, int led_idx, unsigned int *r, unsigned int *g, unsigned int *b)
30 {
31         if (!frame) return;
32         if (!r || !g || !b) return;
33         if (led_idx > frame->num_led) return;
34
35         *r = (frame->leds[led_idx].color & LED_MASK_RED) >> 16;
36         *g = (frame->leds[led_idx].color & LED_MASK_GREEN) >> 8;
37         *b = (frame->leds[led_idx].color & LED_MASK_BLUE);
38 }
39
40 static void
41 _ani_backend_swupdatedone_free_frame(default_frame_info_t *frame)
42 {
43         if (!frame) return;
44
45         if (frame->leds) free(frame->leds);
46         free(frame);
47 }
48
49 static default_frame_info_t *
50 _ani_backend_swupdatedone_get_frame(default_ani_info *ani_info)
51 {
52         default_frame_info_t *frame, *key_frame, *key_frame2;
53         int idx, idx2;
54         unsigned int r, g, b, r2, g2, b2;
55         int r3, g3, b3;
56         double div;
57
58         frame = (default_frame_info_t *)calloc(sizeof(default_frame_info_t), 1);
59         if (!frame) return NULL;
60
61         if (ani_info->frame_idx == 0)
62         {
63                 ani_info->frame_max = (unsigned int)(ani_info->frames[ani_info->key_frame_cur].frame_duration / ani_info->interval);
64         }
65
66         idx = ani_info->key_frame_cur;
67         idx2 = (idx + 1) % ani_info->num_key_frames;
68
69         key_frame = &ani_info->frames[idx];
70         key_frame2 = &ani_info->frames[idx2];
71         frame->num_led = key_frame->num_led;
72         frame->leds = (default_led_info_t *)calloc(sizeof(default_led_info_t), frame->num_led);
73         if (!frame->leds)
74         {
75                 free(frame);
76                 return NULL;
77         }
78
79         div = (double)(ani_info->frame_idx) / (double)ani_info->frame_max;
80         r = g = b = r2 = g2 = b2 = 0x0;
81
82         printf("[jeon] idx: %d, frame_max: %d, frame_idx: %d, div: %lf\n",
83                 idx, ani_info->frame_max, ani_info->frame_idx, div);
84         for (int i = 0; i < key_frame->num_led; i++)
85         {
86                 _ani_backend_swupdatedone_get_led_rgb(key_frame, i, &r, &g, &b);
87                 _ani_backend_swupdatedone_get_led_rgb(key_frame2, i, &r2, &g2, &b2);
88                 r3 = (int)(r2 - r) * div + r;
89                 g3 = (int)(g2 - g) * div + g;
90                 b3 = (int)(b2 - b) * div + b;
91
92                 frame->leds[i].color = (r3 << 16) + (g3 << 8) + (b3);
93                 
94                 printf("[jeon][%d] (%d)rgb(0x%x, 0x%x, 0x%x), (%d)rgb2(0x%x, 0x%x, 0x%x), color: 0x%x(0x%x, 0x%x, 0x%x)\n",
95                         i, idx, r, g, b, idx2, r2, g2, b2, frame->leds[i].color, r3, g3, b3);
96         }
97
98         ani_info->frame_idx++;
99         if (ani_info->frame_idx >= ani_info->frame_max)
100         {
101                 ani_info->frame_idx = 0;
102                 ani_info->key_frame_cur = (ani_info->key_frame_cur + 1) % ani_info->num_key_frames;
103                 if (ani_info->repeat >= 0 && ani_info->key_frame_cur == 0)
104                         ani_info->repeat_cur++;
105         }
106
107         return frame;
108 }
109
110 static pui_bool
111 _ani_backend_swupdatedone_frame_cb(void *data, int serial)
112 {
113         pui_int_error e = PUI_INT_ERROR_NONE;
114         pui_ani_t *ani = (pui_ani_t *)data;
115         pui_backend_ani_data *ani_data = NULL;
116         pui_ani_control_buffer *buffer = NULL;
117         default_frame_info_t *frame;
118         unsigned int r = 0x0, g = 0x0, b = 0x0;
119         double now;
120
121         ani_data = pui_backend_ani_get_ani_data(ani);
122         default_ani_info *ani_info = (default_ani_info *)ani_data->ani_info;
123
124         now = ecore_time_unix_get();
125         pui_info("[time:%.3f] serial=%d\n", now, serial);
126
127         /* TODO : make use of ani_info */
128         //(void) ani_info;
129
130         buffer = pui_backend_ani_get_buffer(ani);
131         if (!buffer) {
132                 pui_err("Failed to get buffer for animation\n");
133                 return (pui_bool)0;
134         }
135
136         frame = _ani_backend_swupdatedone_get_frame(ani_info);
137         for(int i = 0; i<12; i++)
138         {
139                 _ani_backend_swupdatedone_get_led_rgb(frame, i, &r, &g, &b);
140                 buffer->ptr[4*i] = 0;
141                 buffer->ptr[4*i + 1] = b; /* BLUE */
142                 buffer->ptr[4*i + 2] = g; /* GREEN */
143                 buffer->ptr[4*i + 3] = r; /* RED */
144         }
145         _ani_backend_swupdatedone_free_frame(frame);
146
147         e = pui_backend_ani_set_buffer(ani, buffer);
148
149         if (e != PUI_INT_ERROR_NONE)
150         {
151                 pui_err("Failed on setting buffer on animation !(e=%d)\n", e);
152                 return (pui_bool)0;
153         }
154
155         e = pui_backend_ani_update(ani);
156
157         if (e != PUI_INT_ERROR_NONE)
158         {
159                 pui_err("Failed on updating animation !(e=%d)\n", e);
160                 return (pui_bool)0;
161         }
162
163         pui_info("... update (serial=%d), (repeat| cur: %d, want: %d)\n",
164                 serial, ani_info->repeat_cur, ani_info->repeat);
165
166         if (ani_info->repeat >= 0 &&
167                 ani_info->repeat_cur >= ani_info->repeat)
168         {
169                 ani_data->ani_func->ani_stop(ani, EINA_FALSE);
170         }
171
172         return (pui_bool)1;
173 }
174
175 pui_error
176 _ani_swupdatedone_start(pui_ani_t *ani, int repeat)
177 {
178         pui_bool ret = 0;
179         pui_int_error e = PUI_INT_ERROR_NONE;
180         pui_backend_ani_data *ani_data = NULL;
181
182         ani_data = pui_backend_ani_get_ani_data(ani);
183         default_ani_info *info = (default_ani_info *)ani_data->ani_info;
184
185         pui_info("... info->id: %s, repeat : %d, interval: %d\n", info->id, repeat, info->interval);
186
187         pui_backend_ani_status_update(ani, PUI_ANI_STATUS_STARTED);
188         if (repeat == 0) info->repeat = 1;
189         else info->repeat = repeat;
190
191         info->key_frame_cur = 0;
192         info->frame_idx = 0;
193         info->repeat_cur = 0;
194
195         ret = pui_backend_ani_add_frame_cb(ani, _ani_backend_swupdatedone_frame_cb, info->interval / 1000.0);
196         if (!ret)
197         {
198                 pui_err("Failed to add frame callback !\n");
199                 e = PUI_INT_ERROR_INVALID_RESOURCES;
200         }
201
202         return e;
203 }
204
205 pui_error
206 _ani_swupdatedone_stop(pui_ani_t *ani, pui_bool force)
207 {
208         pui_int_error e = PUI_INT_ERROR_NONE;
209         pui_backend_ani_data *ani_data = NULL;
210
211         ani_data = pui_backend_ani_get_ani_data(ani);
212         default_ani_info *info = (default_ani_info *)ani_data->ani_info;
213
214         //TODO
215         (void) info;
216
217         pui_info("... info->id: %s, force=%d\n", info->id, force);
218
219         pui_backend_ani_remove_frame_cb(ani);
220
221         if (force)
222                 pui_backend_ani_status_update(ani, PUI_ANI_STATUS_PAUSED);
223         else
224                 pui_backend_ani_status_update(ani, PUI_ANI_STATUS_STOPPED);
225
226
227         return e;
228 }
229
230
231 void
232 pui_default_backend_ani_swupdatedone_func_set(pui_backend_ani_func *func)
233 {
234         if (!func) return;
235
236         func->ani_start = _ani_swupdatedone_start;
237         func->ani_stop = _ani_swupdatedone_stop;
238 }
239
240