2ff9a846f747cbe9c5cd70cc663145f1fe28bac4
[platform/core/uifw/libpui.git] / backends / voice / default_ani_listening.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_listening_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_listening_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_listening_get_frame(default_ani_info *ani_info)
51 {
52         default_frame_info_t *frame, *key_frame;
53         int division;
54
55         frame = (default_frame_info_t *)calloc(sizeof(default_frame_info_t), 1);
56         if (!frame) return NULL;
57
58         key_frame = &ani_info->frames[0];
59         frame->num_led = key_frame->num_led;
60         frame->leds = (default_led_info_t *)calloc(sizeof(default_led_info_t), frame->num_led);
61         if (!frame->leds)
62         {
63                 free(frame);
64                 return NULL;
65         }
66         division = (int)(frame->num_led / 2);
67         for (int i = 0; i < key_frame->num_led; i++)
68         {
69                 int idx;
70                 //int idx = (i - ani_info->frame_idx + key_frame->num_led) % key_frame->num_led;
71                 if ((ani_info->frame_idx / division) <= 0)
72                 {
73                         if (i < division)
74                         {
75                                 idx = i - ani_info->frame_idx;
76                                 if (idx < 0) continue;
77                         }
78                         else
79                         {
80                                 idx = i + ani_info->frame_idx;
81                                 if (idx >= key_frame->num_led) continue;
82                         }
83                 }
84                 else
85                 {
86                         if (i < division)
87                         {
88                                 idx = i - division + 1 + ani_info->frame_idx - division;
89                                 if (idx < 0) continue;
90                         }
91                         else
92                         {
93                                 idx = i + division - 1 - ani_info->frame_idx + division;
94                                 if (idx >= key_frame->num_led) continue;
95                         }
96                 }
97                 frame->leds[idx] = key_frame->leds[i];
98         }
99         ani_info->frame_idx++;
100         if (ani_info->frame_idx >= key_frame->num_led)
101                 ani_info->frame_idx = 0;
102
103         return frame;
104 }
105
106 static pui_bool
107 _ani_backend_listening_frame_cb(void *data, int serial)
108 {
109         pui_int_error e = PUI_INT_ERROR_NONE;
110         pui_ani_t *ani = (pui_ani_t *)data;
111         pui_backend_ani_data *ani_data = NULL;
112         pui_ani_control_buffer *buffer = NULL;
113         default_frame_info_t *frame;
114         unsigned int r = 0x0, g = 0x0, b = 0x0;
115         double now;
116
117         ani_data = pui_backend_ani_get_ani_data(ani);
118         default_ani_info *ani_info = (default_ani_info *)ani_data->ani_info;
119
120         now = ecore_time_unix_get();
121         pui_info("[time:%.3f] serial=%d\n", now, serial);
122
123         /* TODO : make use of ani_info */
124         //(void) ani_info;
125
126         buffer = pui_backend_ani_get_buffer(ani);
127         if (!buffer) {
128                 pui_err("Failed to get buffer for animation\n");
129                 return (pui_bool)0;
130         }
131
132         frame = _ani_backend_listening_get_frame(ani_info);
133         for(int i = 0; i<12; i++)
134         {
135                 _ani_backend_listening_get_led_rgb(frame, i, &r, &g, &b);
136                 buffer->ptr[4*i] = 0;
137                 buffer->ptr[4*i + 1] = b; /* BLUE */
138                 buffer->ptr[4*i + 2] = g; /* GREEN */
139                 buffer->ptr[4*i + 3] = r; /* RED */
140         }
141         _ani_backend_listening_free_frame(frame);
142
143         e = pui_backend_ani_set_buffer(ani, buffer);
144
145         if (e != PUI_INT_ERROR_NONE)
146         {
147                 pui_err("Failed on setting buffer on animation !(e=%d)\n", e);
148                 return (pui_bool)0;
149         }
150
151         e = pui_backend_ani_update(ani);
152
153         if (e != PUI_INT_ERROR_NONE)
154         {
155                 pui_err("Failed on updating animation !(e=%d)\n", e);
156                 return (pui_bool)0;
157         }
158
159         pui_info("... update (serial=%d)\n", serial);
160
161         return (pui_bool)1;
162 }
163
164 pui_error
165 _ani_listening_start(pui_ani_t *ani, int repeat)
166 {
167         pui_bool ret = 0;
168         pui_int_error e = PUI_INT_ERROR_NONE;
169         pui_backend_ani_data *ani_data = NULL;
170
171         ani_data = pui_backend_ani_get_ani_data(ani);
172         default_ani_info *info = (default_ani_info *)ani_data->ani_info;
173
174         //TODO
175         (void) info;
176
177         pui_info("... info->id: %s, repeat : %d, interval: %d\n", info->id, repeat, info->interval);
178
179         pui_backend_ani_status_update(ani, PUI_ANI_STATUS_STARTED);
180         ret = pui_backend_ani_add_frame_cb(ani, _ani_backend_listening_frame_cb, info->interval / 1000.0);
181
182         if (!ret)
183         {
184                 pui_err("Failed to add frame callback !\n");
185                 e = PUI_INT_ERROR_INVALID_RESOURCES;
186         }
187
188         return e;
189 }
190
191 pui_error
192 _ani_listening_stop(pui_ani_t *ani)
193 {
194         pui_int_error e = PUI_INT_ERROR_NONE;
195         pui_backend_ani_data *ani_data = NULL;
196
197         ani_data = pui_backend_ani_get_ani_data(ani);
198         default_ani_info *info = (default_ani_info *)ani_data->ani_info;
199
200         //TODO
201         (void) info;
202
203         pui_info("... info->id: %s\n", info->id);
204
205         pui_backend_ani_remove_frame_cb(ani);
206         pui_backend_ani_status_update(ani, PUI_ANI_STATUS_STOPPED);
207
208         return e;
209 }
210
211
212 void
213 pui_default_backend_ani_listening_func_set(pui_backend_ani_func *func)
214 {
215         if (!func) return;
216
217         func->ani_start = _ani_listening_start;
218         func->ani_stop = _ani_listening_stop;
219 }
220
221