default_backend: do not allocate frame whenever each frames
[platform/core/uifw/libpui.git] / backends / bt / default_ani_connected.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 default_frame_info_t *ani_frame = NULL;
29
30 static void
31 _ani_backend_connected_get_led_rgb(default_frame_info_t *frame, int led_idx, unsigned int *r, unsigned int *g, unsigned int *b)
32 {
33         if (!frame) return;
34         if (!r || !g || !b) return;
35         if (led_idx > frame->num_led) return;
36
37         *r = (frame->leds[led_idx].color & LED_MASK_RED) >> 16;
38         *g = (frame->leds[led_idx].color & LED_MASK_GREEN) >> 8;
39         *b = (frame->leds[led_idx].color & LED_MASK_BLUE);
40 }
41
42 static default_frame_info_t *
43 _ani_backend_connected_get_frame(default_ani_info *ani_info)
44 {
45         default_frame_info_t *key_frame;
46         int idx;
47
48         if (ani_info->frame_idx == 0)
49         {
50                 ani_info->frame_max = (unsigned int)(ani_info->frames[ani_info->key_frame_cur].frame_duration / ani_info->interval);
51         }
52
53         idx = ani_info->key_frame_cur;
54         key_frame = &ani_info->frames[idx];
55
56         for (int i = 0; i < key_frame->num_led; i++)
57         {
58                 ani_frame->leds[i].color = key_frame->leds[i].color;
59         }
60         ani_info->frame_idx++;
61         if (ani_info->frame_idx >= ani_info->frame_max)
62         {
63                 ani_info->frame_idx = 0;
64                 ani_info->key_frame_cur = (ani_info->key_frame_cur + 1) % ani_info->num_key_frames;
65                 if (ani_info->repeat >= 0 && ani_info->key_frame_cur == 0)
66                         ani_info->repeat_cur++;
67         }
68
69         return ani_frame;
70 }
71
72
73 static pui_bool
74 _ani_backend_connected_frame_cb(void *data, int serial)
75 {
76         pui_int_error e = PUI_INT_ERROR_NONE;
77         pui_ani_t *ani = (pui_ani_t *)data;
78         pui_backend_ani_data *ani_data = NULL;
79         pui_ani_control_buffer *buffer = NULL;
80         default_frame_info_t *frame;
81         unsigned int r = 0x0, g = 0x0, b = 0x0;
82
83         ani_data = pui_backend_ani_get_ani_data(ani);
84         default_ani_info *ani_info = (default_ani_info *)ani_data->ani_info;
85
86         /* TODO : make use of ani_info */
87         //(void) ani_info;
88
89         buffer = pui_backend_ani_get_buffer(ani);
90         if (!buffer) {
91                 pui_err("Failed to get buffer for animation\n");
92                 return (pui_bool)0;
93         }
94
95         frame = _ani_backend_connected_get_frame(ani_info);
96         for(int i = 0; i<12; i++)
97         {
98                 _ani_backend_connected_get_led_rgb(frame, i, &r, &g, &b);
99                 buffer->ptr[4*i] = 0;
100                 buffer->ptr[4*i + 1] = b; /* BLUE */
101                 buffer->ptr[4*i + 2] = g; /* GREEN */
102                 buffer->ptr[4*i + 3] = r; /* RED */
103         }
104         backend_util_cleanup_frame(frame);
105
106         e = pui_backend_ani_set_buffer(ani, buffer);
107
108         if (e != PUI_INT_ERROR_NONE)
109         {
110                 pui_err("Failed on setting buffer on animation !(e=%d)\n", e);
111                 return (pui_bool)0;
112         }
113
114         e = pui_backend_ani_update(ani);
115
116         if (e != PUI_INT_ERROR_NONE)
117         {
118                 pui_err("Failed on updating animation !(e=%d)\n", e);
119                 return (pui_bool)0;
120         }
121
122         pui_info("... update (serial=%d), (repeat| cur: %d, want: %d)\n",
123                 serial, ani_info->repeat_cur, ani_info->repeat);
124
125         if (ani_info->repeat >= 0 &&
126                 ani_info->repeat_cur >= ani_info->repeat)
127         {
128                 ani_data->ani_func->ani_stop(ani, EINA_FALSE);
129         }
130
131         return (pui_bool)1;
132 }
133
134 pui_error
135 _ani_connected_start(pui_ani_t *ani, int repeat)
136 {
137         pui_bool ret = 0;
138         pui_int_error e = PUI_INT_ERROR_NONE;
139         pui_backend_ani_data *ani_data = NULL;
140
141         ani_data = pui_backend_ani_get_ani_data(ani);
142         default_ani_info *info = (default_ani_info *)ani_data->ani_info;
143
144         pui_info("... info->id: %s, repeat : %d, interval: %d\n", info->id, repeat, info->interval);
145
146         pui_backend_ani_status_update(ani, PUI_ANI_STATUS_STARTED);
147         if (repeat == 0) info->repeat = 1;
148         else info->repeat = repeat;
149
150         info->key_frame_cur = 0;
151         info->frame_idx = 0;
152         info->repeat_cur = 0;
153
154         if (!ani_frame)
155                 ani_frame = backend_util_alloc_frame(info);
156         ERROR_CHECK(ani_frame, return PUI_INT_ERROR_INVALID_RESOURCES, "Failed to alloc memory for frame\n");
157
158         ret = pui_backend_ani_add_frame_cb(ani, _ani_backend_connected_frame_cb, info->interval / 1000.0);
159         if (!ret)
160         {
161                 pui_err("Failed to add frame callback !\n");
162                 e = PUI_INT_ERROR_INVALID_RESOURCES;
163         }
164
165         return e;
166 }
167
168 pui_error
169 _ani_connected_stop(pui_ani_t *ani, pui_bool force)
170 {
171         pui_int_error e = PUI_INT_ERROR_NONE;
172         pui_backend_ani_data *ani_data = NULL;
173
174         ani_data = pui_backend_ani_get_ani_data(ani);
175         default_ani_info *info = (default_ani_info *)ani_data->ani_info;
176
177         //TODO
178         (void) info;
179
180         pui_info("... info->id: %s, force=%d\n", info->id, force);
181
182         pui_backend_ani_remove_frame_cb(ani);
183
184         if (force)
185                 pui_backend_ani_status_update(ani, PUI_ANI_STATUS_PAUSED);
186         else
187                 pui_backend_ani_status_update(ani, PUI_ANI_STATUS_STOPPED);
188
189         if (ani_frame)
190         {
191                 backend_util_free_frame(ani_frame);
192                 ani_frame = NULL;
193         }
194
195         return e;
196 }
197
198
199 void
200 pui_default_backend_ani_connected_func_set(pui_backend_ani_func *func)
201 {
202         if (!func) return;
203
204         func->ani_start = _ani_connected_start;
205         func->ani_stop = _ani_connected_stop;
206 }
207
208