blob: adaa543c227fe5b6e9597fde51e8b6782cc3ab5c [file] [log] [blame]
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +09001/*
2 * Copyright (C) 2014 DENSO CORPORATION
3 *
Bryce Harringtonaf637c22015-06-11 12:55:55 -07004 * 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:
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +090011 *
Bryce Harringtonaf637c22015-06-11 12:55:55 -070012 * 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.
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +090024 */
25
26#include <time.h>
27#include <assert.h>
28#include <stdlib.h>
29#include <stdio.h>
Lucas Tanure9b5fe422015-09-21 14:10:32 -030030#include <stdbool.h>
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +090031
32#include "ivi-layout-export.h"
33#include "ivi-layout-private.h"
34
35struct ivi_layout_transition;
36
37typedef void (*ivi_layout_transition_frame_func)(
38 struct ivi_layout_transition *transition);
39typedef void (*ivi_layout_transition_destroy_func)(
40 struct ivi_layout_transition *transition);
41typedef int32_t (*ivi_layout_is_transition_func)(void *private_data, void *id);
42
43struct ivi_layout_transition {
44 enum ivi_layout_transition_type type;
45 void *private_data;
46 void *user_data;
47
48 uint32_t time_start;
49 uint32_t time_duration;
50 uint32_t time_elapsed;
51 uint32_t is_done;
52 ivi_layout_is_transition_func is_transition_func;
53 ivi_layout_transition_frame_func frame_func;
54 ivi_layout_transition_destroy_func destroy_func;
55};
56
57struct transition_node {
58 struct ivi_layout_transition *transition;
59 struct wl_list link;
60};
61
62static void layout_transition_destroy(struct ivi_layout_transition *transition);
63
64static struct ivi_layout_transition *
65get_transition_from_type_and_id(enum ivi_layout_transition_type type,
66 void *id_data)
67{
68 struct ivi_layout *layout = get_instance();
69 struct transition_node *node;
70 struct ivi_layout_transition *tran;
71
72 wl_list_for_each(node, &layout->transitions->transition_list, link) {
73 tran = node->transition;
74
75 if (tran->type == type &&
76 tran->is_transition_func(tran->private_data, id_data))
77 return tran;
78 }
79
80 return NULL;
81}
82
Nobuhiko Tanibataee8e5832014-12-15 13:25:39 +090083int32_t
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +090084is_surface_transition(struct ivi_layout_surface *surface)
85{
86 struct ivi_layout *layout = get_instance();
87 struct transition_node *node;
88 struct ivi_layout_transition *tran;
89
90 wl_list_for_each(node, &layout->transitions->transition_list, link) {
91 tran = node->transition;
92
93 if ((tran->type == IVI_LAYOUT_TRANSITION_VIEW_MOVE_RESIZE ||
94 tran->type == IVI_LAYOUT_TRANSITION_VIEW_RESIZE) &&
95 tran->is_transition_func(tran->private_data, surface))
96 return 1;
97 }
98
99 return 0;
100}
101
102static void
103tick_transition(struct ivi_layout_transition *transition, uint32_t timestamp)
104{
105 const double t = timestamp - transition->time_start;
106
107 if (transition->time_duration <= t) {
108 transition->time_elapsed = transition->time_duration;
109 transition->is_done = 1;
110 } else {
111 transition->time_elapsed = t;
112 }
113}
114
115static float time_to_nowpos(struct ivi_layout_transition *transition)
116{
117 return sin((float)transition->time_elapsed /
118 (float)transition->time_duration * M_PI_2);
119}
120
121static void
122do_transition_frame(struct ivi_layout_transition *transition,
123 uint32_t timestamp)
124{
125 if (0 == transition->time_start)
126 transition->time_start = timestamp;
127
128 tick_transition(transition, timestamp);
129 transition->frame_func(transition);
130
131 if (transition->is_done)
132 layout_transition_destroy(transition);
133}
134
135static int32_t
136layout_transition_frame(void *data)
137{
138 struct ivi_layout_transition_set *transitions = data;
139 uint32_t fps = 30;
140 struct timespec timestamp = {};
141 uint32_t msec = 0;
142 struct transition_node *node = NULL;
143 struct transition_node *next = NULL;
144
145 if (wl_list_empty(&transitions->transition_list)) {
146 wl_event_source_timer_update(transitions->event_source, 0);
147 return 1;
148 }
149
150 wl_event_source_timer_update(transitions->event_source, 1000 / fps);
151
152 clock_gettime(CLOCK_MONOTONIC, &timestamp);/* FIXME */
153 msec = (1e+3 * timestamp.tv_sec + 1e-6 * timestamp.tv_nsec);
154
155 wl_list_for_each_safe(node, next, &transitions->transition_list, link) {
156 do_transition_frame(node->transition, msec);
157 }
158
159 ivi_layout_commit_changes();
160 return 1;
161}
162
Nobuhiko Tanibataee8e5832014-12-15 13:25:39 +0900163struct ivi_layout_transition_set *
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900164ivi_layout_transition_set_create(struct weston_compositor *ec)
165{
166 struct ivi_layout_transition_set *transitions;
167 struct wl_event_loop *loop;
168
169 transitions = malloc(sizeof(*transitions));
170 if (transitions == NULL) {
171 weston_log("%s: memory allocation fails\n", __func__);
172 return NULL;
173 }
174
175 wl_list_init(&transitions->transition_list);
176
177 loop = wl_display_get_event_loop(ec->wl_display);
178 transitions->event_source =
179 wl_event_loop_add_timer(loop, layout_transition_frame,
180 transitions);
181
182 return transitions;
183}
184
Lucas Tanure9b5fe422015-09-21 14:10:32 -0300185static bool
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900186layout_transition_register(struct ivi_layout_transition *trans)
187{
188 struct ivi_layout *layout = get_instance();
189 struct transition_node *node;
190
191 node = malloc(sizeof(*node));
192 if (node == NULL) {
193 weston_log("%s: memory allocation fails\n", __func__);
Lucas Tanure9b5fe422015-09-21 14:10:32 -0300194 return false;
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900195 }
196
197 node->transition = trans;
198 wl_list_insert(&layout->pending_transition_list, &node->link);
Lucas Tanure9b5fe422015-09-21 14:10:32 -0300199 return true;
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900200}
201
202static void
203remove_transition(struct ivi_layout *layout,
204 struct ivi_layout_transition *trans)
205{
206 struct transition_node *node;
207 struct transition_node *next;
208
209 wl_list_for_each_safe(node, next,
210 &layout->transitions->transition_list, link) {
211 if (node->transition == trans) {
212 wl_list_remove(&node->link);
213 free(node);
214 return;
215 }
216 }
217
218 wl_list_for_each_safe(node, next,
219 &layout->pending_transition_list, link) {
220 if (node->transition == trans) {
221 wl_list_remove(&node->link);
222 free(node);
223 return;
224 }
225 }
226}
227
228static void
229layout_transition_destroy(struct ivi_layout_transition *transition)
230{
231 struct ivi_layout *layout = get_instance();
232
233 remove_transition(layout, transition);
Dawid Gajownik74a635b2015-08-06 17:12:19 -0300234 if (transition->destroy_func)
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900235 transition->destroy_func(transition);
236 free(transition);
237}
238
239static struct ivi_layout_transition *
240create_layout_transition(void)
241{
242 struct ivi_layout_transition *transition = malloc(sizeof(*transition));
243
244 if (transition == NULL) {
245 weston_log("%s: memory allocation fails\n", __func__);
246 return NULL;
247 }
248
249 transition->type = IVI_LAYOUT_TRANSITION_MAX;
250 transition->time_start = 0;
251 transition->time_duration = 300; /* 300ms */
252 transition->time_elapsed = 0;
253
254 transition->is_done = 0;
255
John-John Tedro9d7aff02015-09-20 02:47:37 +0200256 transition->is_transition_func = NULL;
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900257 transition->private_data = NULL;
258 transition->user_data = NULL;
259
260 transition->frame_func = NULL;
261 transition->destroy_func = NULL;
262
263 return transition;
264}
265
266/* move and resize view transition */
267
268struct move_resize_view_data {
269 struct ivi_layout_surface *surface;
270 int32_t start_x;
271 int32_t start_y;
272 int32_t end_x;
273 int32_t end_y;
274 int32_t start_width;
275 int32_t start_height;
276 int32_t end_width;
277 int32_t end_height;
278};
279
280static void
281transition_move_resize_view_destroy(struct ivi_layout_transition *transition)
282{
283 struct move_resize_view_data *data =
284 (struct move_resize_view_data *)transition->private_data;
285 struct ivi_layout_surface *layout_surface = data->surface;
286
287 wl_signal_emit(&layout_surface->configured, layout_surface);
288
289 if (transition->private_data) {
290 free(transition->private_data);
291 transition->private_data = NULL;
292 }
293}
294
295static void
296transition_move_resize_view_user_frame(struct ivi_layout_transition *transition)
297{
298 struct move_resize_view_data *mrv = transition->private_data;
299 const double current = time_to_nowpos(transition);
300
301 const int32_t destx = mrv->start_x +
302 (mrv->end_x - mrv->start_x) * current;
303
304 const int32_t desty = mrv->start_y +
305 (mrv->end_y - mrv->start_y) * current;
306
307 const int32_t dest_width = mrv->start_width +
308 (mrv->end_width - mrv->start_width) * current;
309
310 const int32_t dest_height = mrv->start_height +
311 (mrv->end_height - mrv->start_height) * current;
312
313 ivi_layout_surface_set_destination_rectangle(mrv->surface,
314 destx, desty,
315 dest_width, dest_height);
316}
317
318static int32_t
319is_transition_move_resize_view_func(struct move_resize_view_data *data,
320 struct ivi_layout_surface *view)
321{
322 return data->surface == view;
323}
324
325static struct ivi_layout_transition *
326create_move_resize_view_transition(
327 struct ivi_layout_surface *surface,
328 int32_t start_x, int32_t start_y,
329 int32_t end_x, int32_t end_y,
330 int32_t start_width, int32_t start_height,
331 int32_t end_width, int32_t end_height,
332 ivi_layout_transition_frame_func frame_func,
333 ivi_layout_transition_destroy_func destroy_func,
334 uint32_t duration)
335{
Carlos Olmedo Escobar703f5022015-03-02 13:24:36 +0100336 struct ivi_layout_transition *transition;
337 struct move_resize_view_data *data;
338
339 transition = create_layout_transition();
Carlos Olmedo Escobare82ba532015-01-17 19:43:02 +0100340 if (transition == NULL)
341 return NULL;
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900342
Carlos Olmedo Escobar703f5022015-03-02 13:24:36 +0100343 data = malloc(sizeof(*data));
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900344 if (data == NULL) {
345 weston_log("%s: memory allocation fails\n", __func__);
Lucas Tanure9af00172015-09-21 11:24:35 -0300346 free(transition);
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900347 return NULL;
348 }
349
350 transition->type = IVI_LAYOUT_TRANSITION_VIEW_MOVE_RESIZE;
351 transition->is_transition_func = (ivi_layout_is_transition_func)is_transition_move_resize_view_func;
352
353 transition->frame_func = frame_func;
354 transition->destroy_func = destroy_func;
355 transition->private_data = data;
356
357 if (duration != 0)
358 transition->time_duration = duration;
359
360 data->surface = surface;
361 data->start_x = start_x;
362 data->start_y = start_y;
363 data->end_x = end_x;
364 data->end_y = end_y;
365
366 data->start_width = start_width;
367 data->start_height = start_height;
368 data->end_width = end_width;
369 data->end_height = end_height;
370
371 return transition;
372}
373
Nobuhiko Tanibataee8e5832014-12-15 13:25:39 +0900374void
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900375ivi_layout_transition_move_resize_view(struct ivi_layout_surface *surface,
376 int32_t dest_x, int32_t dest_y,
377 int32_t dest_width, int32_t dest_height,
378 uint32_t duration)
379{
380 struct ivi_layout_transition *transition;
381 int32_t start_pos[2] = {
382 surface->pending.prop.start_x,
383 surface->pending.prop.start_y
384 };
385
386 int32_t start_size[2] = {
387 surface->pending.prop.start_width,
388 surface->pending.prop.start_height
389 };
390
391 transition = get_transition_from_type_and_id(
392 IVI_LAYOUT_TRANSITION_VIEW_MOVE_RESIZE,
393 surface);
394 if (transition) {
395 struct move_resize_view_data *data = transition->private_data;
396 transition->time_start = 0;
397 transition->time_duration = duration;
398
399 data->start_x = start_pos[0];
400 data->start_y = start_pos[1];
401 data->end_x = dest_x;
402 data->end_y = dest_y;
403
404 data->start_width = start_size[0];
405 data->start_height = start_size[1];
406 data->end_width = dest_width;
407 data->end_height = dest_height;
408 return;
409 }
410
411 transition = create_move_resize_view_transition(
412 surface,
413 start_pos[0], start_pos[1],
414 dest_x, dest_y,
415 start_size[0], start_size[1],
416 dest_width, dest_height,
417 transition_move_resize_view_user_frame,
418 transition_move_resize_view_destroy,
419 duration);
420
421 layout_transition_register(transition);
422}
423
424/* fade transition */
425struct fade_view_data {
426 struct ivi_layout_surface *surface;
427 double start_alpha;
428 double end_alpha;
429};
430
431struct store_alpha{
432 double alpha;
433};
434
435static void
436fade_view_user_frame(struct ivi_layout_transition *transition)
437{
438 struct fade_view_data *fade = transition->private_data;
439 struct ivi_layout_surface *surface = fade->surface;
440
441 const double current = time_to_nowpos(transition);
442 const double alpha = fade->start_alpha +
443 (fade->end_alpha - fade->start_alpha) * current;
444
445 ivi_layout_surface_set_opacity(surface, wl_fixed_from_double(alpha));
446 ivi_layout_surface_set_visibility(surface, true);
447}
448
449static int32_t
450is_transition_fade_view_func(struct fade_view_data *data,
451 struct ivi_layout_surface *view)
452{
453 return data->surface == view;
454}
455
456static struct ivi_layout_transition *
457create_fade_view_transition(
458 struct ivi_layout_surface *surface,
459 double start_alpha, double end_alpha,
460 ivi_layout_transition_frame_func frame_func,
461 void *user_data,
462 ivi_layout_transition_destroy_func destroy_func,
463 uint32_t duration)
464{
Carlos Olmedo Escobar703f5022015-03-02 13:24:36 +0100465 struct ivi_layout_transition *transition;
466 struct fade_view_data *data;
467
468 transition = create_layout_transition();
Carlos Olmedo Escobare82ba532015-01-17 19:43:02 +0100469 if (transition == NULL)
470 return NULL;
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900471
Carlos Olmedo Escobar703f5022015-03-02 13:24:36 +0100472 data = malloc(sizeof(*data));
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900473 if (data == NULL) {
474 weston_log("%s: memory allocation fails\n", __func__);
Lucas Tanure9af00172015-09-21 11:24:35 -0300475 free(transition);
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900476 return NULL;
477 }
478
479 transition->type = IVI_LAYOUT_TRANSITION_VIEW_FADE;
480 transition->is_transition_func = (ivi_layout_is_transition_func)is_transition_fade_view_func;
481
482 transition->user_data = user_data;
483 transition->private_data = data;
484 transition->frame_func = frame_func;
485 transition->destroy_func = destroy_func;
486
487 if (duration != 0)
488 transition->time_duration = duration;
489
490 data->surface = surface;
491 data->start_alpha = start_alpha;
492 data->end_alpha = end_alpha;
493
494 return transition;
495}
496
497static void
498create_visibility_transition(struct ivi_layout_surface *surface,
499 double start_alpha,
500 double dest_alpha,
501 void *user_data,
502 ivi_layout_transition_destroy_func destroy_func,
503 uint32_t duration)
504{
505 struct ivi_layout_transition *transition = NULL;
506
507 transition = create_fade_view_transition(
508 surface,
509 start_alpha, dest_alpha,
510 fade_view_user_frame,
511 user_data,
512 destroy_func,
513 duration);
514
Lucas Tanurec8dcd162015-09-23 10:33:21 -0300515 if (!layout_transition_register(transition))
Lucas Tanure96db0772015-09-21 14:10:33 -0300516 layout_transition_destroy(transition);
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900517}
518
519static void
520visibility_on_transition_destroy(struct ivi_layout_transition *transition)
521{
522 struct fade_view_data *data = transition->private_data;
523 struct store_alpha *user_data = transition->user_data;
524
525 ivi_layout_surface_set_visibility(data->surface, true);
526
527 free(data);
528 transition->private_data = NULL;
529
530 free(user_data);
531 transition->user_data = NULL;
532}
533
Nobuhiko Tanibataee8e5832014-12-15 13:25:39 +0900534void
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900535ivi_layout_transition_visibility_on(struct ivi_layout_surface *surface,
536 uint32_t duration)
537{
538 struct ivi_layout_transition *transition;
539 bool is_visible = ivi_layout_surface_get_visibility(surface);
540 wl_fixed_t dest_alpha = ivi_layout_surface_get_opacity(surface);
541 struct store_alpha *user_data = NULL;
542 wl_fixed_t start_alpha = 0.0;
543 struct fade_view_data *data = NULL;
544
545 transition = get_transition_from_type_and_id(
546 IVI_LAYOUT_TRANSITION_VIEW_FADE,
547 surface);
548 if (transition) {
549 start_alpha = ivi_layout_surface_get_opacity(surface);
550 user_data = transition->user_data;
551 data = transition->private_data;
552
553 transition->time_start = 0;
554 transition->time_duration = duration;
555 transition->destroy_func = visibility_on_transition_destroy;
556
557 data->start_alpha = wl_fixed_to_double(start_alpha);
558 data->end_alpha = user_data->alpha;
559 return;
560 }
561
562 if (is_visible)
563 return;
564
565 user_data = malloc(sizeof(*user_data));
566 if (user_data == NULL) {
567 weston_log("%s: memory allocation fails\n", __func__);
568 return;
569 }
570
571 user_data->alpha = wl_fixed_to_double(dest_alpha);
572
573 create_visibility_transition(surface,
574 0.0, // start_alpha
575 wl_fixed_to_double(dest_alpha),
576 user_data,
577 visibility_on_transition_destroy,
578 duration);
579}
580
581static void
582visibility_off_transition_destroy(struct ivi_layout_transition *transition)
583{
584 struct fade_view_data *data = transition->private_data;
585 struct store_alpha *user_data = transition->user_data;
586
587 ivi_layout_surface_set_visibility(data->surface, false);
588
589 ivi_layout_surface_set_opacity(data->surface,
590 wl_fixed_from_double(user_data->alpha));
591
592 free(data);
593 transition->private_data = NULL;
594
595 free(user_data);
596 transition->user_data= NULL;
597}
598
Nobuhiko Tanibataee8e5832014-12-15 13:25:39 +0900599void
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900600ivi_layout_transition_visibility_off(struct ivi_layout_surface *surface,
601 uint32_t duration)
602{
603 struct ivi_layout_transition *transition;
604 wl_fixed_t start_alpha = ivi_layout_surface_get_opacity(surface);
605 struct store_alpha* user_data = NULL;
606 struct fade_view_data* data = NULL;
607
608 transition =
609 get_transition_from_type_and_id(IVI_LAYOUT_TRANSITION_VIEW_FADE,
610 surface);
611 if (transition) {
612 data = transition->private_data;
613
614 transition->time_start = 0;
615 transition->time_duration = duration;
616 transition->destroy_func = visibility_off_transition_destroy;
617
618 data->start_alpha = wl_fixed_to_double(start_alpha);
619 data->end_alpha = 0;
620 return;
621 }
622
623 user_data = malloc(sizeof(*user_data));
624 if (user_data == NULL) {
625 weston_log("%s: memory allocation fails\n", __func__);
626 return;
627 }
628
629 user_data->alpha = wl_fixed_to_double(start_alpha);
630
631 create_visibility_transition(surface,
632 wl_fixed_to_double(start_alpha),
633 0.0, // dest_alpha
634 user_data,
635 visibility_off_transition_destroy,
636 duration);
637}
638
639/* move layer transition */
640
641struct move_layer_data {
642 struct ivi_layout_layer *layer;
643 int32_t start_x;
644 int32_t start_y;
645 int32_t end_x;
646 int32_t end_y;
647 ivi_layout_transition_destroy_user_func destroy_func;
648};
649
650static void
651transition_move_layer_user_frame(struct ivi_layout_transition *transition)
652{
653 struct move_layer_data *data = transition->private_data;
654 struct ivi_layout_layer *layer = data->layer;
655
656 const float current = time_to_nowpos(transition);
657
658 const int32_t dest_x = data->start_x +
659 (data->end_x - data->start_x) * current;
660
661 const int32_t dest_y = data->start_y +
662 (data->end_y - data->start_y) * current;
663
664 ivi_layout_layer_set_position(layer, dest_x, dest_y);
665}
666
667static void
668transition_move_layer_destroy(struct ivi_layout_transition *transition)
669{
670 struct move_layer_data *data = transition->private_data;
671
Dawid Gajownik74a635b2015-08-06 17:12:19 -0300672 if (data->destroy_func)
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900673 data->destroy_func(transition->user_data);
674
675 free(data);
676 transition->private_data = NULL;
677}
678
679static int32_t
680is_transition_move_layer_func(struct move_layer_data *data,
681 struct ivi_layout_layer *layer)
682{
683 return data->layer == layer;
684}
685
686
687static struct ivi_layout_transition *
688create_move_layer_transition(
689 struct ivi_layout_layer *layer,
690 int32_t start_x, int32_t start_y,
691 int32_t end_x, int32_t end_y,
692 void *user_data,
693 ivi_layout_transition_destroy_user_func destroy_user_func,
694 uint32_t duration)
695{
Carlos Olmedo Escobar703f5022015-03-02 13:24:36 +0100696 struct ivi_layout_transition *transition;
697 struct move_layer_data *data;
698
699 transition = create_layout_transition();
Carlos Olmedo Escobare82ba532015-01-17 19:43:02 +0100700 if (transition == NULL)
701 return NULL;
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900702
Carlos Olmedo Escobar703f5022015-03-02 13:24:36 +0100703 data = malloc(sizeof(*data));
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900704 if (data == NULL) {
705 weston_log("%s: memory allocation fails\n", __func__);
Lucas Tanure9af00172015-09-21 11:24:35 -0300706 free(transition);
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900707 return NULL;
708 }
709
710 transition->type = IVI_LAYOUT_TRANSITION_LAYER_MOVE;
711 transition->is_transition_func = (ivi_layout_is_transition_func)is_transition_move_layer_func;
712
713 transition->frame_func = transition_move_layer_user_frame;
714 transition->destroy_func = transition_move_layer_destroy;
715 transition->private_data = data;
716 transition->user_data = user_data;
717
718 if (duration != 0)
719 transition->time_duration = duration;
720
721 data->layer = layer;
722 data->start_x = start_x;
723 data->start_y = start_y;
724 data->end_x = end_x;
725 data->end_y = end_y;
726 data->destroy_func = destroy_user_func;
727
728 return transition;
729}
730
Nobuhiko Tanibataee8e5832014-12-15 13:25:39 +0900731void
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900732ivi_layout_transition_move_layer(struct ivi_layout_layer *layer,
733 int32_t dest_x, int32_t dest_y,
734 uint32_t duration)
735{
736 int32_t start_pos_x = 0;
737 int32_t start_pos_y = 0;
738 struct ivi_layout_transition *transition = NULL;
739
740 ivi_layout_layer_get_position(layer, &start_pos_x, &start_pos_y);
741
742 transition = create_move_layer_transition(
743 layer,
744 start_pos_x, start_pos_y,
745 dest_x, dest_y,
746 NULL, NULL,
747 duration);
748
Lucas Tanurec8dcd162015-09-23 10:33:21 -0300749 if (transition && layout_transition_register(transition))
750 return;
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900751
Lucas Tanurec8dcd162015-09-23 10:33:21 -0300752 free(transition);
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900753}
754
Nobuhiko Tanibataee8e5832014-12-15 13:25:39 +0900755void
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900756ivi_layout_transition_move_layer_cancel(struct ivi_layout_layer *layer)
757{
758 struct ivi_layout_transition *transition =
759 get_transition_from_type_and_id(
760 IVI_LAYOUT_TRANSITION_LAYER_MOVE,
761 layer);
762 if (transition) {
763 layout_transition_destroy(transition);
764 }
765}
766
767/* fade layer transition */
768struct fade_layer_data {
769 struct ivi_layout_layer *layer;
770 uint32_t is_fade_in;
771 double start_alpha;
772 double end_alpha;
773 ivi_layout_transition_destroy_user_func destroy_func;
774};
775
776static void
777transition_fade_layer_destroy(struct ivi_layout_transition *transition)
778{
779 struct fade_layer_data *data = transition->private_data;
780 transition->private_data = NULL;
781
782 free(data);
783}
784
785static void
786transition_fade_layer_user_frame(struct ivi_layout_transition *transition)
787{
788 double current = time_to_nowpos(transition);
789 struct fade_layer_data *data = transition->private_data;
790 double alpha = data->start_alpha +
791 (data->end_alpha - data->start_alpha) * current;
792 wl_fixed_t fixed_alpha = wl_fixed_from_double(alpha);
793
794 int32_t is_done = transition->is_done;
795 bool is_visible = !is_done || data->is_fade_in;
796
797 ivi_layout_layer_set_opacity(data->layer, fixed_alpha);
798 ivi_layout_layer_set_visibility(data->layer, is_visible);
799}
800
801static int32_t
802is_transition_fade_layer_func(struct fade_layer_data *data,
803 struct ivi_layout_layer *layer)
804{
805 return data->layer == layer;
806}
807
Nobuhiko Tanibataee8e5832014-12-15 13:25:39 +0900808void
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900809ivi_layout_transition_fade_layer(
810 struct ivi_layout_layer *layer,
811 uint32_t is_fade_in,
812 double start_alpha, double end_alpha,
813 void* user_data,
814 ivi_layout_transition_destroy_user_func destroy_func,
815 uint32_t duration)
816{
817 struct ivi_layout_transition *transition;
818 struct fade_layer_data *data = NULL;
819 wl_fixed_t fixed_opacity = 0.0;
820 double now_opacity = 0.0;
821 double remain = 0.0;
822
823 transition = get_transition_from_type_and_id(
824 IVI_LAYOUT_TRANSITION_LAYER_FADE,
825 layer);
826 if (transition) {
827 /* transition update */
828 data = transition->private_data;
829
830 /* FIXME */
831 fixed_opacity = ivi_layout_layer_get_opacity(layer);
832 now_opacity = wl_fixed_to_double(fixed_opacity);
833 remain = 0.0;
834
835 data->is_fade_in = is_fade_in;
836 data->start_alpha = now_opacity;
837 data->end_alpha = end_alpha;
838
839 remain = is_fade_in? 1.0 - now_opacity : now_opacity;
840 transition->time_start = 0;
841 transition->time_elapsed = 0;
842 transition->time_duration = duration * remain;
843
844 return;
845 }
846
847 transition = create_layout_transition();
Carlos Olmedo Escobare82ba532015-01-17 19:43:02 +0100848 if (transition == NULL)
849 return;
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900850
Carlos Olmedo Escobare82ba532015-01-17 19:43:02 +0100851 data = malloc(sizeof(*data));
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900852 if (data == NULL) {
853 weston_log("%s: memory allocation fails\n", __func__);
Lucas Tanurec8dcd162015-09-23 10:33:21 -0300854 free(transition);
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900855 return;
856 }
857
858 transition->type = IVI_LAYOUT_TRANSITION_LAYER_FADE;
859 transition->is_transition_func = (ivi_layout_is_transition_func)is_transition_fade_layer_func;
860
861 transition->private_data = data;
862 transition->user_data = user_data;
863
864 transition->frame_func = transition_fade_layer_user_frame;
865 transition->destroy_func = transition_fade_layer_destroy;
866
867 if (duration != 0)
868 transition->time_duration = duration;
869
870 data->layer = layer;
871 data->is_fade_in = is_fade_in;
872 data->start_alpha = start_alpha;
873 data->end_alpha = end_alpha;
874 data->destroy_func = destroy_func;
875
Lucas Tanurec8dcd162015-09-23 10:33:21 -0300876 if (!layout_transition_register(transition))
877 layout_transition_destroy(transition);
Nobuhiko Tanibata6f9df652014-11-27 13:22:00 +0900878
879 return;
880}
881